C++之指针与引用,函数和数组
- 引用:是对象的别名,其内部存放的时一个对象的地址,通过引用可以操纵对象,引用的所有操作实际上都是应用在它所指的对象身上,包括取地址操作符。引用的一些特性:
- 引用定义时必须被初始化,引用一旦定义就不能再指向其他对象
- const引用可以用不同类型的对象初始化,只要能从这种类型转换到引用类型,对于不可寻址的值如常量,编译器为了实现引用会生成一个临时对象,引用实际指向该对象,非const引用不能指向需要临时对象的对象或值,会导致编译错误
- 不能定义引用类型的数组
- 从上一点可以看出引用内部会存放一个内存地址,所以引用也是占内存的,但不能改变它的指向
- 函数返回引用时要注意,引用不能指向函数的局部变量,因为在函数执行完后局部变量内存会被清空,所以引用的值会没有意义
- 指针:指针变量所存的内容时它指向的内存地址,用法有:
- TYPE *ptr:定义一个指向TYPE的指针
- TYPE *ptr[n]:定义一个指向TYPE的指针的数组
- TYPE (*ptr)[n]:定义一个指向数组的指针,数组元素数目为n.ptr中存放的其实时数组第一个元素的地址,所以取数组中第一元素要用**ptr,两次解指针操作
- int arr[3]={1,2,3};
- //ptr是指针,该指针类型是int[3]
- int (*ptr)[3]=&arr;
cout << **ptr << endl;//第一次解指针时得到数组地址,第二次解指针取数组中的值
- int arr[3]={1,2,3};
- TYPE **ptr:定义一个指向TYPE类型的指针的指
- 空类型指针(void *)可以被任何类型数据指针类型的地址赋值,包括函数指针.由于不知道该地址的数据类型,所以不能操作空类型指针所指的对象,只能传送该地址或与其他地址值比较
- int add(int a,int b)
- {return a+b;}
- void *ptr = &add;//vs 2012编译通过
- int add(int a,int b)
- 指针与函数:
- TYPE (*ptr)(int,int):定义一个指向函数的指针,函数返回类型为TYPE,包含两个int参数。函数指针不是类型安全的,不同签名的函数指针之间能够强制转换,编译器能通过,但执行时候会出现不可预期的行为. 函数名其实时指向该类型函数的指针,将取地址符作用于函数名也能得到函数地址,如下add与&add是相同类型
- void add(int a,int b)
- {return ;}
- typedef int (*FunPtr)(int a,int b,int c);
- FunPtr func =(FunPtr)add;//编译能通过,执行时行为不可预期
- void add(int a,int b)
- 函数指针的数组:int(*Func[10])(int,int),Func是一个拥有10个元素的数组,每个元素都是一个指向函数的函数指针,这种定义方式可读性较差,可以使用如下方式
- typedef (*FuncPtr)(int,int)
- FuncPtr Func[10]
- typedef (*FuncPtr)(int,int)
- 返回类型为函数指针的函数:
- int(*func(int))(int *,int)//声明 一个函数func,返回类型为int (*)(int*,int)
- typedef int (*FuncPtr)(int *,int)//使用typedef提高代码可读性
- FuncPtr func(int *,int)
- 要注意的是函数名不能作为函数的返回类型,只有函数指针可以,另外函数名作为参数传递给函数时,也会被转换为函数指针,对函数名也不能使用解指针操作符
- TYPE (*ptr)(int,int):定义一个指向函数的指针,函数返回类型为TYPE,包含两个int参数。函数指针不是类型安全的,不同签名的函数指针之间能够强制转换,编译器能通过,但执行时候会出现不可预期的行为. 函数名其实时指向该类型函数的指针,将取地址符作用于函数名也能得到函数地址,如下add与&add是相同类型
- 指针与引用的区别:
- 引用不能为空,指针可以
- sizeof 引用得到的是变量的大小,sizeof指针得到的是指针本身的大小,任何指针类型在32位系统都是4字节
- 引用时类型安全的,指针不是
- 指针与数组的关系:
- 数组与指针是两种不同类型,数组有明确数量,指针只是个地址值.
- 单独使用数组名时,编译器会把数组名转换为一个指针常量,指向数组中第一个元素的地址。对于二维数组数组名则是指针的指针
- int arr[3]={1,2,3}
- //如下语句作用一样
- int *ptr = arr;
- int *ptr=&arr[0]
- int arr[2][3]={{1,2,3},{4,5,6}}
- int (*ptr)[3];//定义一个指向数组的指针,数组元素数目为3
- ptr = &arr[1];
- cout << **p << endl 输出4,注意是两次解指针操作
- 对于二维数组,数组名其实是一个指针,可以叫做行指针,该指针指向一个数组的首地址,数组元素数目为二维数组的列数.所以数组名也可以认为是指针的指针.
- int arr[2][3]={{1,2,3},{4,5,6}};
- cout <<"arr: "<< arr << endl;
- cout <<"arr+1: " << arr+1 << endl;//第二行的地址
- cout << "*arr: " << *arr << endl;
- cout << "*(arr+1): " << *(arr+1) << endl;//第二行的首地址
- cout <<"arr[0]: " << arr[0] << endl;
- cout <<"&arr[0]: " << &arr[0] << endl;
- cout <<"&arr[0]+1: " << &arr[0]+1 << endl;
- cout <<"arr[1]: " << arr[1] << endl;
- cout <<"&arr[1]: " << &arr[0]+1 << endl;
- int arr[2][3]={{1,2,3},{4,5,6}};
C++之指针与引用,函数和数组的更多相关文章
- 数组类型与sizeof与指针的引用
以char类型为例: char a[100]; //a类型为char[100] &a类型为 char (*)[100] *a类型为char char *p = a; ...
- 动态创建二维vector数组 C和C++ 及指针与引用的区别
二维vectorvector<vector <int> > ivec(m ,vector<int>(n)); //m*n的二维vector 动态创建m*n的二 ...
- 通过数组初始化链表的两种方法:指向指针的引用node *&tail和指向指针的指针(二维指针)node **tail
面试高频题:单链表的逆置操作/链表逆序相关文章 点击打开 void init_node(node *tail,char *init_array) 这样声明函数是不正确的,函数的原意是通过数组初始化链表 ...
- C++基础回顾2(函数, 指针和引用)
接着回顾函数.指针和应用. 函数 1.多维数组作为形参时,第一维的大小可以省略(也可以不省略),但是其他维的大小必须指定.比如二维数组形参,int array[3][]不正确,int arry[][1 ...
- (C/C++)区别:数组与指针,指针与引用
1.数组跟指针的区别 数组要么在静态存储区被创建(如全局数组),要么在栈上被创建.数组名对应着(而不是指向)一块内存,其地址与容量在生命期内保持不变,只有数组的内容可以改变. 指针可以随时指向任意类型 ...
- C++中指针和引用、数组之间的区别
指针指向一块内存,它的内容是所指内存的地址:而引用则是某块内存的别名,引用初始化后不能改变指向.使用时,引用更加安全,指针更加灵活. 初始化.引用必须初始化,且初始化之后不能呢改变:指针可以不必初始化 ...
- C++之指针,引用与数组
引用只是对象的另一个名字,通过在变量名前面添加"&”符号来定义,而指针保存的是另一个对象的地址,它们两都提供了间接访问所服务变量的途径. 但是它们的差别还是挺大的: 先从它们的值说起 ...
- <c和指针>学习笔记3之函数和数组
1 函数声明 (1)原型 告诉编译器函数的参数数量和每个参数的类型以及返回值的类型.编译器通过检查原型之后,就可以检查这个函数得调用,从而来确保参数正确,返回值无误. 通用技巧,将原型写在一个头文件当 ...
- PHP中使用数组指针函数操作数组示例
数组的内部指针是数组内部的组织机制,指向一个数组中的某个元素.默认是指向数组中第一个元素通过移动或改变指针的位置,可以访问数组中的任意元素.对于数组指针的控制PHP提供了以下几个内建函数可以利用. ★ ...
随机推荐
- js接收文件流并下载
js接收文件流并下载 标签(空格分隔): js 在此输入正文 <script type="text/javascript"> function download(fil ...
- 为IT程序员量身定制的12个目标——很经典
对程序员们来说挑战自我非常重要,要么不断创新,要么技术停滞不前.新年伊始,我整理了12个月的目标,每个目标都是对技术或个人能力的挑战,而且可以年复一年循环使用. 01. 变得有耐心 02. 保持健康 ...
- caffe mnist实例 --lenet_train_test.prototxt 网络配置详解
1.mnist实例 ##1.数据下载 获得mnist的数据包,在caffe根目录下执行./data/mnist/get_mnist.sh脚本. get_mnist.sh脚本先下载样本库并进行解压缩,得 ...
- 日志记录~log4.net
1. 添加Log4net引用 2. 添加配置文件 Log.config <?xml version="1.0" encoding="utf-8"?> ...
- C#操作IIS服务
进入正题:先从使用角度来讲解IIS操作,然后再深入到具体的IIS服务底层原理. [1]前提掌握要点: (1).IIS到目前经历了四个版本分别为 IIS4.0 IIS5.0 IIS6.0 IIS7.0, ...
- css样式表格边框1px hover时为2px 实现方式
//css .flclass-cont .flclass-cont-box{width:%;display:inline-block;font-size:;margin:10px;position:r ...
- Unity 手机屏幕翻转问题 横屏
1920*1080的图在1080*1920的设备上观看效果: 如果要做横屏游戏,就要改Build中的Player Settings,强制左旋转或右旋转,默认是Auto 垂直于地面的手机在横屏下分辨率是 ...
- protocol 和delegate(协议和代理)的区别
定义 protocol:中文叫协议,一个只有方法体(没有具体实现)的类,Java中称作接口,实现协议的类必须实现协议中@required标记的方法(如果有的话): delegate:中文叫代理或委托, ...
- Python 进行网络编程
Date: 2019-06-10 Author: Sun 1. Python TCP通信实现 socket()函数 Python 中,我们用 socket()函数来创建套接字,语法格式如下: sock ...
- Python的流程控制
条件判断 通过`if`,`elif`,`else`关键字来实现条件判断逻辑的实现,执行改结构中的其中一个,其结构如下: if condition1: pass elif condition2: pas ...