指针和动态内存分配

数组与指针

数组

  • 数组名是一个指针常量。
  • 数组名传递数据时,传递的是地址。
  • 数组作为函数参数时不指定第一维大小。
  • 对象数组

A a[2] = {A(1,2)};

执行时先调用有参数的构造函数初始化a[0],然后调用默认构造函数初始化a[1]。

指针

  1. 指向常量的指针 。不能改变所指对象的值,但可以指向其他对象。
    int a;
const int* p1 = &a;
int b;
p1 = &b;//正确,可以指向其他对象。
*p1 = 1;//错误,不能改变所指对象的值
  1. 指针类型的常量 。指针本身的值不能被改变。
    int a,b;
int* const p1 = &a;
p1 = &b; //错误,指针本身的值不能被改变

指针数组

    int line1[] = {1,0,0};
int line2[] = {0,1,0};
int line3[] = {0,0,1};
int* PLine[] = {line1,line2,line3};
for(int i = 0;i < 3;++i){
for(int j = 0;j < 3;++j)
cout<<PLine[i][j]<<" ";
cout<<endl;
}

!

函数指针

void foo(int a){
cout<<"cout foo"<<a<<endl;
}
void bar(int b){
cout<<"cout bar"<<b<<endl;
}
int main()
{
void (*ptr)(int);
ptr = foo;
ptr(1);
ptr = bar;
ptr(2);
return 0;
}

对象指针

指向对象的指针

Point* ptr;
Point p1;
ptr = &p1;//指向一个对象
p->getX(); //(*p).getX() 调用相应的成员
this指针

this指针是隐含于每一个类的非静态成员函数中的特殊指针,用于指向正在被成员函数操作的对象。

动态内存分配

一般对象的动态内存分配

在C++中建立和删除堆对象使用的两个操作符:new和delete。

*new操作 *

该语句运行过程中申请分配用于存放指定类型数据的内存空间,并根据初始化参数列表中的值进行初始化。如果内存申请成功,便 返回一个指向新分配内存首地址的类型的指针 。可以通过该指针对堆对象进行访问。如果申请失败则抛出异常。

  • 分配基本类型的变量。
    int* p = new int; //分配内存后不设初值。
int* p1 = new int();//分配内存后,用0来初始化。
int* p1 = new int(1);
  • 分配类类型变量
    class Point {
public:
Point():x(0),y(0){
cout<<"Default constructor called."<<endl;
}
Point(int x,int y):x(x),y(y){
cout<<"Constructor called."<<endl;
}
~Point(){cout<<"Destructor called."<<endl;}
private:
int x,y;
};
int main()
{
cout<<"Step one"<<endl;
Point* ptr1 = new Point;//未给出参数列表,调用默认ctor
delete ptr1;//删除对象,调用dtor cout<<"Step two"<<endl;
ptr1 = new Point(1,2);//给出参数列表,调用有形ctor
delete ptr1;//删除对象,调用dtor
return 0;
}

注意:

  1. 用new建立一个类类型的变量时,如果该类存在用户定义的默认ctor,则“new T”和“new T()”的效果是相同的。
  2. 若用户没有定义默认的ctor,使用“new T”创建对象时,会调用系统隐含的ctor;使用“new T()”创建对象时,系统除了执行那些操作外,还会为基本数据类型和指针类型赋0,而且过程是递归的。

delete操作

如果被删除的是对象,对象的析构函数将被调用。用new建立的对象,只能用delete删除一次。

动态数组

一维数组

int* arr = new int[n]; //new T
int* arr = new int [n]() ;//new T()
int* arr = new int [n](1) ;//用1初始化。
delete[] arr;//删除整个数组。

多维数组的分配

/*
new 类型名T[第一维][第二维]…
分配一个2*3*4的三位数组的两种方法。
1.整个分配,释放的时候是整个释放掉。
2.一层一层分配,释放的时候也是从内而外一层一层释放。
*/
int(*cp)[3][4];
cp = new int [2][3][4]();
delete cp; //分配空间
int*** p = new int**[2];
for(int i = 0;i < 2; ++i){
p[i] = new int*[3];
for(int j = 0;j < 3; ++j)
p[i][j] = new int[4]();
}
//赋值
for(int i = 0;i < 2; ++i)
for(int j = 0;j < 3; ++j)
for(int k = 0;k < 4; ++k)
p[i][j][k] = i*j*k;
//释放空间
for(int i = 0;i < 2; ++i){
for(int j = 0;j < 3; ++j)
delete[] p[i][j];
delete[] p[i];
}
delete[] p;

深拷贝和浅拷贝

隐含的拷贝构造函数完成的是浅拷贝。

浅拷贝与深拷贝的区别如图。

class Point{
...
};
class ArrayOfPoint{
public:
ArrayOfPoint(const ArrayOfPoint& v);
private:
Point* points;
int size;
};
ArrayOfPoint::ArrayOfPoint(const ArrayOfPoint& v)
{
//深拷贝,为新的对象重新开辟一片内存空间。
size = v.size;
points = new Point[size];
for(int i = 0;i < size; ++i)
points[i] = v.points[i];
}

C++指针和动态内存分配的更多相关文章

  1. 数据结构基础——指针及动态内存分配(malloc)

    一.指针 C语言中的指针是一种数据类型,比如说我们用int *a;就定义了一个指针a,它指向一个int类型的数.但是这个指针是未初始化的,所以,一般的,我们都在创建指针时初始化它,以免出错,在还不吃的 ...

  2. c++ 动态数组,指针与动态内存分配

    教学内容: 内存的使用 动态内存分配malloc函数 分配内存时使用sizeof运算符 用指针访问内存 以数组的形式访问内存 一.内存的使用 堆(heap) 在程序执行期间分配内存时,内存区域中的这个 ...

  3. C++学习 之 指针及动态内存分配(笔记)

    1. 指针 1.1 指针的含义: 简单来说,指针是存储内存地址的变量.当我们声明了一个指针变量后,使用该指针时系统会根据指针内的地址索引内存块,读取内存内的值.指针因为是用来存地址的,所以一般固定长度 ...

  4. C++二维数组动态内存分配

    对于二维数组和二维指针的内存的分配 这里首选说一下一维指针和一维数组的内存分配情况. 一维: 数组:形如int  a[5];这里定义了一个一维数组a,并且数组的元素个数是5,这里的a是这五个元素的整体 ...

  5. C和指针 第十一章 动态内存分配

    声明数组时,必须指定数组长度,才可以编译,但是如果需要在运行时,指定数组的长度的话,那么就需要动态的分配内存. C函数库stdlib.h提供了两个函数,malloc和free,分别用于执行动态内存分配 ...

  6. 《C和指针》 读书笔记 -- 第11章 动态内存分配

    1.C函数库提供了两个函数,malloc和free,分别用于执行动态内存分配和释放,这些函数维护一个可用内存池. void *malloc(size_t size);//返回指向分配的内存块起始位置的 ...

  7. C动态内存分配(C与指针实例)

    主要初步介绍malloc.free.calloc.realloc的基本.日后会有更详细的内容. malloc.free分别用于动态内存分配和释放. malloc会从内存池里提取一块合适的内存(连续的) ...

  8. C++学习笔记(十一):void*指针、类型转换和动态内存分配

    void*指针 void关键字表示“空类型”的概念.但是,这里的“空类型”不表示“任意类型”,而是表示不存在的意思,也就是说C/C++不允许你写语句void a,不存在类型为void的东西. void ...

  9. C和C指针小记(十六)-动态内存分配

    动态内存分配 1.1 为什么使用动态内存分配 直接声明数组的方式的缺点: 1) 声明数组必须指定长度限制.无法处理超过声明长度的数组. 2) 如果声明更大的常量来弥补第一个缺点,会造成更多的内存浪费. ...

随机推荐

  1. springMVC4 注解配置实例

    结构: maven配置: <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http:// ...

  2. 使用pip安装BeautifulSoup4模块

    1.测试是否安装了BeautifulSoup4模块 import bs4 print bs4 执行报错说明没有安装该模块 Traceback (most recent call last): File ...

  3. 【转】Polya定理

    转自:http://endlesscount.blog.163.com/blog/static/82119787201221324524202/ Polya定理 首先记Sn为有前n个正整数组成的集合, ...

  4. 另类的表单数据"序列化"

    背景:最近在做项目时,由于编辑页面需要提交的数据量有些多,而且在提交前还需要做一些逻辑处理,所以如果按照正常的方式,一个个的获取值然后拼接json对象传到后台的话相对工作量较大,而且容易出错,后期的维 ...

  5. activity与fragment之间传递数据

    总结:无论是activity给fragment传递数据,还是fragment给activity传递数据,都把activity和fragment都当做一个普通的对象,调用它的方法,传递参数. 1.Fra ...

  6. IOS第二天-新浪微博 - 添加搜索框,弹出下拉菜单 ,代理的使用 ,HWTabBar.h(自定义TabBar)

    ********HWDiscoverViewController.m(发现) - (void)viewDidLoad { [super viewDidLoad]; // 创建搜索框对象 HWSearc ...

  7. 动态Web API层

    返回总目录 本篇目录 构建动态Web API控制器 ForAll 方法 重写ForAll 方法 Http动词 动态Javascript代理 Ajax参数 单一服务脚本 Angular支持 Durand ...

  8. 飞鱼(FlyFish)——便捷的原型在线制作工具

    关于项目原型制作,小菜先前写过一篇文章<FastUI快速界面原型制作工具>,只不过那个是用C#写的原型制作工具,但是感觉用C#写起来比较费力,而且也不太好用,经过高人指点,茅塞顿开,决定重 ...

  9. 再探.NET的PE文件结构(安全篇)

    一.开篇 首先写在前面,这篇文章源于个人的研究和探索,由于.NET有自己的反射机制,可以清楚的将源码反射出来,这样你的软件就很容易被破解,当然这篇文章不会说怎么样保护你的软件不被破解,相反是借用一个软 ...

  10. 过段时间逐步使用HTML5新增的web worker等内容

    想来快2017年了,2013年前的手机应该很少有人用了,以后逐渐使用HTML5新增的高级API吧. 先把web worker的内容再熟悉一下,因为微软虚拟学院的'面向有经验开发人员的 JavaScri ...