C/C++ 内存分配 new 操作符:剖析new操作符的实现机制和使用技巧

简介: C/C++ 内存分配 new 操作符:剖析new操作符的实现机制和使用技巧

概述


C++是一种面向对象编程语言,它提供了一系列的内存管理工具,其中最重要的就是new操作符。new是用来动态分配内存的,它能够在程序运行时创建一个新的对象,并返回一个指向该对象的指针。
在C++中,使用new操作符可以分配两种类型的内存:堆内存和自由存储区。堆内存是在程序运行时动态分配的内存,通常用于存储较大的数据结构,如数组和对象。自由存储区是程序运行时维护的一段内存区域,可以用来存储较小的数据结构,如指针和变量。

new和operator new


new为C++中的一个关键字(准确地说是操作符)。
operator new是一个C++库函数


  • new操作符执行以下操作
  1. operator new(unsigned int) ------ 分配内存空间
  2. Demo::Demo() -------- 调用适当的构造函数在上面申请的空间上构建对象,更改 指针的属性 .new

  • operator new执行以下操作
  1. 分配所需大小的内存空间。
  2. 返回指向分配内存空间的指针。

总的来说,new 在分配内存后会自动调用对象的构造函数,而 operator new 只是为分配内存提供了一种方法。在大多数情况下,C++程序员通常使用 new 而不是 operator new,因为 new 提供了更方便的语法,并自动处理对象的构造和析构。

new 重载格式

  • 以成员函数的形式重载 new 运算符
void * className::operator new( size_t size ){
    //TODO: }

  • 以全局函数的形式重载 new 运算符
void * operator new( size_t size ){
    //TODO: }
void * opertor new (size_t size ,const char * file ,long line) 
{ 
cout << file << “:” << line << endl; 
 cout << "这是第" <<i++<<"次使用new操作符"<< endl;
void *p = malloc(size); 
return p 
} 
void operator delete(void* ptr)
{
    std::cout << "Deallocating memory" << std::endl;
    std::free(ptr);
}
void * operator new(size_t i)//new的重载示例
{
        cout << "使用new操作符" << endl;
        if (i > sizeof(A))//如果调用此new的是派生类(大于A)
        {
            return ::operator new(i);//那么调用全局的正式new
        }
        return malloc(i);
}

原则:

  • 只分配所要求的空间,不调用相关对象的构造函数。当无法满足所要求分配的空间时,则如果有new_handler,则调用new_handler,否则如果没要求不抛出异常(以nothrow参数表达),则执行bad_alloc异常,否则返回回0 .
  • 可以被重载
  • 重载时,返回类型必须声明为void*
  • 重载时,第一个参数类型必须为表达要求分配空间的大小(字节),类型为size_t
  • 重载时,可以带其它参数

源码

operator new源码

//头文件先忽略
using std::new_handler;
using std::bad_alloc;// using 声明
#if _GLIBCXX_HOSTED// 如果 有这个宏使用 std的malloc
using std::malloc;
#else// 没有则使用c语言的malloc
// A freestanding C runtime may not provide "malloc" -- but there is no
// other reasonable way to implement "operator new".
extern "C" void *malloc (std::size_t);
#endif
_GLIBCXX_WEAK_DEFINITION void *
operator new (std::size_t sz) _GLIBCXX_THROW (std::bad_alloc)
{
  void *p;
  // new_handler 以后说明,但是可以看出首先我们根据入口参数 sz的大小分配内存, 
  // 如果sz为0 则令其为1 ,然后在while循环中调用malloc申请内存
  // 直到 申请成功 或者 抛出异常或者 abort
  /* malloc (0) is unpredictable; avoid it.  */
  if (sz == 0)
    sz = 1;
  while (__builtin_expect ((p = malloc (sz)) == 0, false))
    {
      new_handler handler = std::get_new_handler ();
      if (! handler)
    _GLIBCXX_THROW_OR_ABORT(bad_alloc());
      handler ();
    }
  return p;
}

placement new(特殊的new操作)(不分配内存 + 构造函数的调用)

// Default placement versions of operator new.
inline void* operator new(std::size_t, void* __p) _GLIBCXX_USE_NOEXCEPT
{ return __p; }
//可以看到 placement new 什么内存都没有申请 只是直接返回 指针 那么placement new的步骤和new的步骤像只是没有申请任何的内存但是调用了构造函数
//当我们使用 `obj p = new( alreadyExistPoint )obj()` 意味着在已经分配内存的 内容里调用obj的构造函数 那么步骤是这样的 
//调用 重载的 operator new 函数 
void * tmp_point = ::operator new( sizeof( obj ) ,alreadyExistPoint) ; 
// 更改 指针的属性 
obj * point = static_cast

array new源码

_GLIBCXX_WEAK_DEFINITION void*
operator new[] (std::size_t sz) _GLIBCXX_THROW (std::bad_alloc)
{
  return ::operator new(sz);
}

new和malloc的区别

  • 语法:new 是 C++ 的运算符,而 malloc 是 C 标准库函数。因此,new 可以调用类的构造函数,而 malloc 不行。
  • 返回类型:new 返回指向已分配的内存空间的指针,并且可以自动确定所需内存的大小,而 malloc 返回 void*,需要手动指定分配内存的大小。
  • 异常处理:如果 new 无法分配所需的内存空间,它将抛出 std::bad_alloc 异常,而 malloc 只会返回一个空指针。
  • 用法:new 可以使用在任何需要分配对象内存的情况下,malloc 则更适用于动态数组的分配。
  • 内存对齐:new 会自动保证返回的指针所指向的内存是按照所需类型的对齐方式对齐的,而 malloc 则需要手动调用一些特定的函数才能确保对齐。

在使用时,通常在 C++ 中使用 new,在 C 中使用 malloc。

operator new和malloc的区别

operator new 和 malloc 都是用于动态分配内存的函数。它们之间的主要区别如下:

  • operator new 是 C++ 中的操作符函数,而 malloc 是 C 标准库中的函数。
  • operator new 用于在堆上分配特定类型的内存,而 malloc 分配的是一段无类型的内存空间。
  • operator new 会自动调用构造函数初始化对象,而 malloc 不会。
  • operator new 可以被用户自定义,从而实现定制的内存分配行为,而 malloc 不支持这样的定制。
  • operator new 还支持重载,可以根据需要重载不同版本的 operator new 来实现不同的内存分配策略,而 malloc 不支持重载。
  • operator new 和 malloc 返回的指针类型不同,operator new 返回的指针类型是所请求类型的指针,而 malloc 返回的是 void* 类型的指针,需要手动转换为所需的类型。

在 C++ 中,通常使用 operator new 来分配内存,特别是在需要为对象类型分配内存时,而在 C 中,则通常使用 malloc。

目录
相关文章
|
5月前
|
存储 监控 算法
基于 C++ 哈希表算法实现局域网监控电脑屏幕的数据加速机制研究
企业网络安全与办公管理需求日益复杂的学术语境下,局域网监控电脑屏幕作为保障信息安全、规范员工操作的重要手段,已然成为网络安全领域的关键研究对象。其作用类似网络空间中的 “电子眼”,实时捕获每台电脑屏幕上的操作动态。然而,面对海量监控数据,实现高效数据存储与快速检索,已成为提升监控系统性能的核心挑战。本文聚焦于 C++ 语言中的哈希表算法,深入探究其如何成为局域网监控电脑屏幕数据处理的 “加速引擎”,并通过详尽的代码示例,展现其强大功能与应用价值。
116 2
|
3月前
|
安全 C语言 C++
比较C++的内存分配与管理方式new/delete与C语言中的malloc/realloc/calloc/free。
在实用性方面,C++的内存管理方式提供了面向对象的特性,它是处理构造和析构、需要类型安全和异常处理的首选方案。而C语言的内存管理函数适用于简单的内存分配,例如分配原始内存块或复杂性较低的数据结构,没有构造和析构的要求。当从C迁移到C++,或在C++中使用C代码时,了解两种内存管理方式的差异非常重要。
129 26
|
6月前
|
缓存 并行计算 PyTorch
PyTorch CUDA内存管理优化:深度理解GPU资源分配与缓存机制
本文深入探讨了PyTorch中GPU内存管理的核心机制,特别是CUDA缓存分配器的作用与优化策略。文章分析了常见的“CUDA out of memory”问题及其成因,并通过实际案例(如Llama 1B模型训练)展示了内存分配模式。PyTorch的缓存分配器通过内存池化、延迟释放和碎片化优化等技术,显著提升了内存使用效率,减少了系统调用开销。此外,文章还介绍了高级优化方法,包括混合精度训练、梯度检查点技术及自定义内存分配器配置。这些策略有助于开发者在有限硬件资源下实现更高性能的深度学习模型训练与推理。
1032 0
|
8月前
|
存储 程序员 编译器
玩转C++内存管理:从新手到高手的必备指南
C++中的内存管理是编写高效、可靠程序的关键所在。C++不仅继承了C语言的内存管理方式,还增加了面向对象的内存分配机制,使得内存管理既有灵活性,也更加复杂。学习内存管理不仅有助于提升程序效率,还有助于理解计算机的工作原理和资源分配策略。
|
4月前
|
C语言 C++
c与c++的内存管理
再比如还有这样的分组: 这种分组是最正确的给出内存四个分区名字:栈区、堆区、全局区(俗话也叫静态变量区)、代码区(也叫代码段)(代码段又分很多种,比如常量区)当然也会看到别的定义如:两者都正确,记那个都选,我选择的是第一个。再比如还有这样的分组: 这种分组是最正确的答案分别是 C C C A A A A A D A B。
64 1
|
7月前
|
存储 Linux C语言
C++/C的内存管理
本文主要讲解C++/C中的程序区域划分与内存管理方式。首先介绍程序区域,包括栈(存储局部变量等,向下增长)、堆(动态内存分配,向上分配)、数据段(存储静态和全局变量)及代码段(存放可执行代码)。接着探讨C++内存管理,new/delete操作符相比C语言的malloc/free更强大,支持对象构造与析构。还深入解析了new/delete的实现原理、定位new表达式以及二者与malloc/free的区别。最后附上一句鸡汤激励大家行动缓解焦虑。
|
8月前
|
存储 监控 算法
公司监控上网软件架构:基于 C++ 链表算法的数据关联机制探讨
在数字化办公时代,公司监控上网软件成为企业管理网络资源和保障信息安全的关键工具。本文深入剖析C++中的链表数据结构及其在该软件中的应用。链表通过节点存储网络访问记录,具备高效插入、删除操作及节省内存的优势,助力企业实时追踪员工上网行为,提升运营效率并降低安全风险。示例代码展示了如何用C++实现链表记录上网行为,并模拟发送至服务器。链表为公司监控上网软件提供了灵活高效的数据管理方式,但实际开发还需考虑安全性、隐私保护等多方面因素。
118 0
公司监控上网软件架构:基于 C++ 链表算法的数据关联机制探讨
|
8月前
|
安全 C语言 C++
彻底摘明白 C++ 的动态内存分配原理
大家好,我是V哥。C++的动态内存分配允许程序在运行时请求和释放内存,主要通过`new`/`delete`(用于对象)及`malloc`/`calloc`/`realloc`/`free`(继承自C语言)实现。`new`分配并初始化对象内存,`delete`释放并调用析构函数;而`malloc`等函数仅处理裸内存,不涉及构造与析构。掌握这些可有效管理内存,避免泄漏和悬空指针问题。智能指针如`std::unique_ptr`和`std::shared_ptr`能自动管理内存,确保异常安全。关注威哥爱编程,了解更多全栈开发技巧。 先赞再看后评论,腰缠万贯财进门。
350 0
|
9月前
|
存储 程序员 编译器
什么是内存泄漏?C++中如何检测和解决?
大家好,我是V哥。内存泄露是编程中的常见问题,可能导致程序崩溃。特别是在金三银四跳槽季,面试官常问此问题。本文将探讨内存泄露的定义、危害、检测方法及解决策略,帮助你掌握这一关键知识点。通过学习如何正确管理内存、使用智能指针和RAII原则,避免内存泄露,提升代码健壮性。同时,了解常见的内存泄露场景,如忘记释放内存、异常处理不当等,确保在面试中不被秒杀。最后,预祝大家新的一年工作顺利,涨薪多多!关注威哥爱编程,一起成为更好的程序员。
339 0
|
8月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。

热门文章

最新文章