指针类型的函数:函数的返回值是指针。

不要将非静态局部地址用作函数的返回值,离开函数后就失效了

在子函数中定义局部变量后将其地址返回给函数就是非法地址

在子函数中用new操作取得的内存地址返回给主函数合法有效,不会自动消失,必须用delete。

函数指针的用途——函数回调:将函数指针用作参数传递给另一个函数,例子如下

int compare (int a ,int b ,int (*fun)(int,int))

{return fun(a,b);}

int max(int a, int b)

{return ((a>b)>?a:b);}

int min(int a,int b)

{return ((a>b)>?a:b);}

函数指针充当不同的函数名,调用相应的函数功能

-------------------------------------------------------------

this指针:隐含于类的每个非静态成员函数中,参数列表为空,实际传了当前对象的地址给this,不然返回的时候不知道给谁

-----------------------------------------------------------

当不知道处理数据规模有多大,也不知道数组为多大合适,直到程序运行起来才确定,这时候就需要动态分配内存单元,动态分配的内存单元,来不及起变量名,没有数组名,只返回一个首地址。

分配和释放动态数组:

new    类型名T  [数组长度]  // 数组长度可以是任何整数类型表达式,在运行时计算

delete  []   指针名 //删除的是指针所指对象,不是删除指针,指针还可以赋值 

delete pi; //这是释放了pi所指的目标的内存空间,也就是撤销了目标,但指针pi并没有撤销,它依然存在,该指针所占内存空间并未释放

int *pi=new int (0);  大致等价 int val=0; *pi=&val;  区别就是在堆中分配对象没有名字,不会自动初始化,必须用括号中的值初始化该对象

delete是如何知道要为多少个对象调用析构函数的: 

new在分配内存时,对于需要调用析构函数的类,会多分配对应类型的字节数,如:int就分配4个字节来保存对象的个数,所以在delete的时候括号中不用写明释放大小。

char (*fp)[3]; //二维数组,这个fp是指向整个数组的指针,fp+1是越过整个行

接受一个动态分配多维数组首地址的指针,要定义一个指向多维数组的指针,去掉第一维的下标个数,后面的都留着。

--------------------------------------------------------

智能指针:

unique_ptr:不允许多个指针共享资源,可以用标准库中的move函数转移指针

shared_ptr:多个指针共享资源

weak_ptr:可复制shared_ptr,但其构造或释放对资源不产生影响

-------------------------------------------------------

当定义一个封装动态数组的类,把内存空间分配和释放都在类中定义,使用更安全方便,存在的问题,自己封装的动态数组,只能容纳一个数据类型

解决办法:采用vector,能容纳任何类型的数组,只需要使用时告诉它类型即可,自动进行内存空间分配和删除,并进行下标越界检查

vector是个模板,vector<double> 就生成一个double这么一个类,这个类专门用来存放double类型元素,在vector中已经重载了下标运算符。

-------------------------------------------------------

有的时候复制构造是不必要的,可以通过移动解决

因此有了移动语义,将源对象资源的控制权全部交给目标对象

针对:临时对象即将消亡,并且里面的资源仍旧需要

实现方法:   用参数对应的指针对象来初始化当前对象的指针,然后把参数对象指针置为空指针,析构函数delete一个空指针不发生任何事情

即将消亡的值和函数返回的临时变量是右值。

-----------------------------------------------------

使用动态分配内存的数组,需要自己定义一个类,并且要定义构造和析构函数,每个类只能存放特定一种类型

用vector模板类,其中已经定义好了构造,析构函数,不用自己定义,而且只需要使用时先告诉它需要定义一个什么类对象<>传入即可,vector中可以存放任何类数据类型。

-----------------------------------------------------

new和delete是管理堆内存的

new的过程:

用关键字new在堆上动态创建一个对象,只不过返回指向首地址的指针,总共做了三件事:

1,获得一块内存空间(大小由类型决定)

2,调用构造函数——在创建简单类型变量时这步可省略

3,返回正确的指针

new失败时会调用分配内存失败处理程序

delete是如何知道要为多少个对象调用析构函数的:new分配内存空间时,针对需要调用析构函数的类,多分配对应类型的字节数,如:int就分配4个字节来保存对象的个数,所以在delete的时候括号中不用写明释放大小。

---------------------------------------------------------

new operator,就是平时用的new

operator new:像加减乘除一样可以被重载,步骤:首先调用分配内存的代码,尝试得到一段堆上的空间,成功则返回,失败去调用new_handle重复前面的过程

placement new: 定位构造,实现new的第二步,在取得一块可容纳指定类型对象的内存后,在这块内存上构造对象,使用时必须包含头文件new,并且可以在栈上构造,可以在堆或者栈上构造。

-------------------------------------------------------

new()分配这种类型的一个大小的内存空间,并用括号中的值来初始化这个变量

new[] 分配这种类型的n个大小的内存空间,并用默认构造函数来初始化这些变量

operator new:是个全局函数,写在一个文件中,当使用new关键字时,编译器会自动找到这个函数并调用这个函数

全局主要是因为每个类还可以重载自己的operator new() 函数

malloc用在内置数据类型或结构时,能满足需求,用在类类型时不能

new即分配堆内存,又自动调用类的构造函数来创建对象

new(申请内存时还调用构造函数)和delete(释放内存前,会调用析构函数),是运算符,对应C中的malloc(只会申请内存)和free(只会释放内存),是函数,

每个new获得对象后,必须用delete释放

C++98中,new创建的对象数组不能被显式初始化,数组所有元素被确省初始化,数组元素类型必须有构造函数

C++11中允许显式初始化: int *p_int=new int[3] {1,2,3}

引自:https://blog.csdn.net/hihozoo/article/details/51441521

this指针 new 和delete的更多相关文章

  1. c/c++ 复习基础要点01-const指针、指针函数 函数指针、new/delete与malloc/free区别与联系

    1.      引用本身是有指针实现的:引用为只读指针 例子: int d=123; int& e=d;    //引用 int * const e=d; //只读指针,e指向d,不可修改e指 ...

  2. [C++ Primer Plus] 第4章、复合类型(一)程序清单——指针new和delete

    程序清单4.1 #include<iostream> using namespace std; void main(){ ]; yams[]=; yams[]=; yams[]=; ]={ ...

  3. 转载:指针delete后要设置为NULL

    本文来自:http://rpy000.blog.163.com/blog/static/196109536201292615547939/ 众所周知,最开始我们用new来创建一个指针,那么等我们用完它 ...

  4. delete指针以后应赋值为NULL——QT deletelater指针以后也同样要马上赋值为NULL

    delete p后,只是释放了指针指向的内存空间.p并不会自动被置为NULL,而且指针还在,同时还指向了之前的地址 delete NULL编译器不会报错(因为delete空指针是合法的) 例: 对一个 ...

  5. c++ 用new后delete,而继续输出指针后果 new/new[]/delete/delete[]区别

    #include<iostream> #include<cstring> #include <string.h> int main(){ ]; ;i<;i++ ...

  6. C++指针delete后还要置为null

    非常好的一篇说明: 转载:https://blog.csdn.net/qq_36570733/article/details/80043321 众所周知,最开始我们用new来创建一个指针,那么等我们用 ...

  7. 写了placement new就要写placement delete

    "placement new"通常是专指指定了位置的new(std::size_t size, void *mem),用于vector申请capacity剩余的可用内存. 但广义的 ...

  8. iOS开发_内存泄漏、内存溢出和野指针之间的区别

    今天,在工作群中,被问到了内存泄漏和野指针指向的区别,自己答的不是很好,特意回来查了资料,在博文中总结一下经验,欢迎指正. 内存泄漏:是指在堆区,alloc 或new 创建了一个对象,但是并没有放到自 ...

  9. C++提前delete

    ////////////////////////////////////// ///类析构以后,成员变量内存空间释放, ///函数 和 变量 还是可以引用的 ///////////////////// ...

随机推荐

  1. @PathVariable 与@RequestParam

    http://localhost:8080/Springmvc/user/page.do?pageSize=3&pageNow=2 你可以把这地址分开理解,其中问号前半部分:http://lo ...

  2. 基于ceph快照快速回滚openstack上的虚拟机

    查看虚拟机ID 1 2 [root@node1 ~]# nova list --all | grep wyl | dc828fed-1c4f-4e5d-ae84-795a0e71eecc | wyl ...

  3. HTML各种标签复习

    <html>      --开始标签 <head> 网页上的控制信息 <title>页面标题</title> </head> <bod ...

  4. 立个FLAG

    今天再次初步浏览了寒假生活: 三篇阅读笔记(人月神话,梦断代码,程序员修炼之道:从小工到专家),2月24日之前发表开发记账本软件,连续七天以上发表开发进度 学会使用GitHub,录制抖音(父母评价作品 ...

  5. C 中的typedef应用

    1. typedef  声明的新的类型名在变量名的位置出现. example: typedef unsigned int UINT 则 unsigned int a; 相当于 UINT A; 2. t ...

  6. CentOS 6.5下Redis安装测试

    NoSQL之Redis - CentOS 6.5安装测试 1.下载redis 可以在线安装或者下载 redis ①在线安装前需要检测是否存在rpm包不存在的话查看yum在线是否存在rpm包不存在的话就 ...

  7. VS2015 添加web服务(电子看板)

    鼠标右键点击项目,选择引用----添加服务引用,输入WS地址后单击转到 http://ERPPRD02:8000/sap/bc/srt/wsdl/flv_10002A111AD1/bndg_url/s ...

  8. js中switch/case分支的值可以是变量或表达式

    在一些高级语言如C#中,switch分支的值只能是常量,而js中可以是变量或表达式: <!DOCTYPE html> <html lang="en"> &l ...

  9. Eclipse kepler 安装 Dynamic Web Project差距WTP

    原文地址:http://blog.csdn.net/westrain2010/article/details/25122999, 欢迎转载 Eclipse 标准版是不能创建 Dynamic Web P ...

  10. ss源码学习--事件处理

    为了方便区分,以下分别使用local,server,remote代表ss客户端,ss服务端,以及ss客户端请求访问的远程主机. 在shadowsocks中,无论对于local还是server,都需要建 ...