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。

目录
相关文章
|
1月前
|
存储 监控 算法
Java中的内存管理:理解Garbage Collection机制
本文将深入探讨Java编程语言中的内存管理,着重介绍垃圾回收(Garbage Collection, GC)机制。通过阐述GC的工作原理、常见算法及其在Java中的应用,帮助读者提高程序的性能和稳定性。我们将从基本原理出发,逐步深入到调优实践,为开发者提供一套系统的理解和优化Java应用中内存管理的方法。
|
12天前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
35 4
|
15天前
|
存储 安全 编译器
【c++】深入理解别名机制--引用
本文介绍了C++中的引用概念及其定义、特性、实用性和与指针的区别。引用是C++中的一种别名机制,通过引用可以实现类似于指针的功能,但更安全、简洁。文章详细解释了引用的定义方式、引用传参和返回值的应用场景,以及常引用的使用方法。最后,对比了引用和指针的异同,强调了引用在编程中的重要性和优势。
30 1
|
22天前
|
存储 算法 Java
Go语言的内存管理机制
【10月更文挑战第25天】Go语言的内存管理机制
22 2
|
24天前
|
存储 运维 Java
💻Java零基础:深入了解Java内存机制
【10月更文挑战第18天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
29 1
|
1月前
|
存储 程序员 编译器
简述 C、C++程序编译的内存分配情况
在C和C++程序编译过程中,内存被划分为几个区域进行分配:代码区存储常量和执行指令;全局/静态变量区存放全局变量及静态变量;栈区管理函数参数、局部变量等;堆区则用于动态分配内存,由程序员控制释放,共同支撑着程序运行时的数据存储与处理需求。
106 21
|
1月前
|
程序员 C++ 容器
在 C++中,realloc 函数返回 NULL 时,需要手动释放原来的内存吗?
在 C++ 中,当 realloc 函数返回 NULL 时,表示内存重新分配失败,但原内存块仍然有效,因此需要手动释放原来的内存,以避免内存泄漏。
|
1月前
|
存储 C语言 C++
【C++打怪之路Lv6】-- 内存管理
【C++打怪之路Lv6】-- 内存管理
38 0
【C++打怪之路Lv6】-- 内存管理
|
1月前
|
C++
C/C++内存管理(下)
C/C++内存管理(下)
49 0
|
1月前
|
存储 Linux C语言
C/C++内存管理(上)
C/C++内存管理(上)
38 0