C++的new&delete

new的过程

new的过程:先分配memory,再调用ctor

我们常用的创建对象的方法有两种

Complex c(1,2);                 //栈
Complex *pc = new Complex(1,2); //堆

第一种创建出来的对象将保存在栈上,第二种则在堆上,必须手动回收内存空间(通过delete)

为了解释new的过程,我们先建立一个Complex类

class Complex
{
public:
    Complex(...) {...}//构造函数
    ...
private:
    double real;
    double imag;
};

当我们使用new构建Complex类对象的时候

Complex *pc = new Complex(1,2);

当我们使用new这一个动作,在上动态创建一个对象时,编译器实际上帮你做了三件事:

Complex *pc;

//1.分配内存
void* memory = operator new(sizeof(Complex));
//2.转型
pc = static_cast<Complex*>(memory);
//3.调用构造函数
pc->Complex::Complex(1,2);
  1. 分配内存:operator new也是一个函数,其内部调用malloc(n),拿到sizeof(Complex)大小的内存空间;这时候我们得到指向内存空间始址的指针memory,它是一个指向viod类型的指针
  2. 转型:用static_cast函数,把步骤①得到的指针memory(这是一个pointer to void)转换为pointer to Complex,并将其赋值到pc(步骤①和②可以写在一起)
  3. 调用构造函数:步骤②得到的指针pc指向的内存空间,即为新对象的起始内存地址;于是编译器将通过指针pc调用对象的构造函数

所以从结果上看,这两段代码是等效的

//代码1.
Complex *pc = new Complex(1,2);
//代码2.
Complex *pc;
void* memory = operator new(sizeof(Complex));
pc = static_cast<Complex*>(memory);
pc->Complex::Complex(1,2);

malloc和new的区别在于,当malloc失败时,它不会调用分配内存失败处理程序new_handler,因此我们还是要尽可能的使用new,除非有一些特殊的需求

delete的过程

delete的过程:先调用dtor,再释放memory

我们再建立一个包含指针的类String:

class String {
public:
    ...
    ~String()
    {delete[] m_data;}
    ...
private:
    char* m_data;
};

当我们试用new&delete时:

String* ps = new String("HELLO");
...
delete ps;

编译器在delete这里实际上帮你做了两件事:

String::~String(ps);    //1.调用析构函数
operator delete(ps);    //2.释放内存
  1. 调用析构函数:由于String类是包含指针的,所以设计时不能使用默认析构函数,而是重载一个符合需求的析构函数,在我们delete ps时,编译器第一步就是调用我们重载后的析构函数(没有重载则调用默认)
  2. 释放内存:operator deleteoperator new一样也是一个函数,其内部调用free(ps)

new的三种形态

有的朋友可能被上面的newoperator new搞晕了,实际上在C++中提到new,至少可能代表以下三种含义:new operatoroperator newplacement new

new operator

我们上面所说的new,都是指new operator,也就是我们平时使用的new

operator new

new operator的第一步分配内存是通过调用operator new来完成的,这里的“new”实际上是像加减乘除一样的操作符,因此也是可以重载的

operator new默认情况下首先调用分配内存的代码,尝试得到一段堆上的空间,如果成功就返回,如果失败,则转而去调用一个new_hander,然后继续重复前面过程

如果我们对这个过程不满意,就可以重载operator new,来设置我们希望的行为,例如在Complex类里加入:

class Complex
{
public:
    Complex(...) {...}//构造函数
    ...
    void* operator new(size_t size){
        printf("operator new called\n");
        //通过::operator new调用了原有的全局的new
        return ::operator new(size);
    }
private:
    double real;
    double imag;
};

这里通过::operator new调用了原有的全局的new,在分配内存之前输出一句话

delete也有delete operatoroperator delete之分,后者也是可以重载的。并且,如果重载了operator new,就应该也相应的重载operator delete,这是良好的编程习惯。

placement new(坑)

placement new是用来实现定位构造的,因此可以实现new operator三步操作中的调用构造函数这一步(在取得了足够内存空间后,在这块内存空间是上构造一个对象)

上面写的pc->Complex::Complex(1,2);这句话并不是一个标准的写法,正确的写法是使用placement new

#include <new.h>

原文地址:https://www.cnblogs.com/zhxmdefj/p/12292669.html

时间: 2024-11-02 23:29:30

C++的new&delete的相关文章

Hbase delete遇到的常见异常: Exception in thread &quot;main&quot; java.lang.UnsupportedOperationException

hbase 执行批量删除时出现错误: Exception in thread "main" java.lang.UnsupportedOperationException at java.util.AbstractList.remove(AbstractList.java:161) at org.apache.hadoop.hbase.client.HTable.delete(HTable.java:852) 这种异常其实很常见,remove操作不支持,为什么会出现不支持的情况呢?检查

mariadb 实用功能2 delete from table returning ... 数据恢复

做为一名dba,每天都是和数据大交到,每天也都会从后台来执行一些sql,原则应该是在前端来操作, 但是总是有很多很多的特殊情况.每次执行sql的时候都需要做备份,防止误操作. 看看mariadb的一个小功能吧,又是超级贴心的,我是非常喜欢的.删除数据的时候,可以把删除 显示出来.不多说了,看操作,敲起来!! MariaDB [test]> select version(); +-----------------+ | version()       | +-----------------+ |

oracle中delete drop truncate的用法和区别

数据库的运维中,经常会遇到delete drop truncate的操作,那么如何去把握它们的用法和区别呢? 比如当数据库空间爆满,已经增长到存储空间单个存储文件的最大值32G.你需要通过一些办法释放掉表空间或者扩容表空间来解决问题. 一般当系统中大量使用分区表,而针对分区表清除数据,是不会释放表空间的,必须把分区drop掉,才会释放空间. 下面我们具体了解一下这三个命令: 一.delete 1.delete是DML,执行delete操作时,每次从表中删除一行,并且同时将该行的的删除操作记录在r

237. Delete Node in a Linked List

1. 问题描述 Write a function to delete a node (except the tail) in a singly linked list, given only access to that node. Supposed the linked list is 1 -> 2 -> 3 -> 4 and you are given the third node with value 3, the linked list should become 1 ->

Maven Failed to execute goal org.apache.maven.plugins:maven-clean-plugin:2.5:clean Failed to delete access_log

I'm trying to run simple struts project using maven and tomcat. When I'm trying to exucute next goals:clean install tomcat7:run ,there is an error : [ERROR] Failed to execute goal org.apache.maven.plugins:maven-clean-plugin:2.5:clean (default-clean)

没有躲过的坑--成对使用new和delete时要采取相同的形式

new创建类对象与不new区别: new创建类对象需要指针接收,一处初始化,多处使用 new创建类对象使用完需delete销毁 new创建对象直接使用堆空间,而局部不用new定义类对象则使用栈空间 new对象指针用途广泛,比如作为函数返回值.函数参数等 而且每个学习C++编程的人都知道成对的使用new和delete,也也就是new申请的内存用delete释放,new []申请的内存由delete []释放. std::string* first_string = new std::string;

mac 中vmware fusion 使用技巧 (delete and control)

在MAC上使用虚拟机时,CTRL+左键=右键也对使用WORD非常不便.解决的方法除了楼上的使用Karabiner工具将此系统快捷键重定向之外,还有一种非常简便的方法,即在虚拟机状态下使用CTRL+COMMAND+左键,其效果等同于纯PC下的CTRL+左键. 按住[option]键再按[delete]可以删除光标前的一句话或一整个单词按住[command]再按[delete],可以删除一整行文字macbook上的向后删除(就像PC上键盘的delete),是按住[fn]再按[delete]

delete与delete [] 真正区别

我们通常从教科书上看到这样的说明: delete 释放new分配的单个对象指针指向的内存 delete[] 释放new分配的对象数组指针指向的内存 那么,按照教科书的理解,我们看下下面的代码: int *a = new int[10]; delete a;        //方式1 delete [] a;     //方式2 肯定会有很多人说方式1肯定存在内存泄漏,是这样吗? 1. 针对简单类型 使用new分配后的不管是数组还是非数组形式内存空间用两种方式均可 如: int *a = new

eclipse工程总是提示红叉,但是没有看到哪出错了!The project was not built due to &quot;Could not delete

最近在编译web project的时候,页面总是提示xx方法没有被定义为xx类.但是明明都是正确的. 而且在工程上有个红叉但是程序并没有看到哪有错误.如下图: 解决方法: 1.先打开problem窗口,才能看到意想不到的错误. Window-show view-other-找到problem.然后就会看到一条错误.如下: The project was not built due to "Could not delete '/build/com'.". Fix the problem,

Es6中如何使用splic,delete等数组删除方法

Es6中如何使用splic,delete等数组删除方法 1:js中的splice方法 splice(index,len,[item])    注释:该方法会改变原始数组. splice有3个参数,它也可以用来替换/删除/添加数组内某一个或者几个值 index:数组开始下标        len: 替换/删除的长度       item:替换的值,删除操作的话 item为空 如:arr = ['a','b','c','d'] 删除 ----  item不设置 arr.splice(1,1)   /