初学C++时,很容易把指针和引用的用法混在一起,下面通过一些示例来说明指针和引用两者之间的差别。
1、两者的定义和性质不同
指针是一个变量,存储的是一个地址,指向内存的一个存储单元;
引用是原变量的一个别名,跟原来的变量实质上是同一个东西。
Plain Text int a = 996; int *p = &a; // p是指针, &在此是求地址运算 int &r = a; // r是引用, &在此起标识作用 |
上面定义了一个整型变量 a,p 是一个指针变量,p 的值是变量 a 的地址;
而引用 r,是 a 的一个别名,在内存中 r 和 a 占有同一个存储单元。
2、指针可以有多级,引用只能是一级
Plain Text int **p; // 合法 int &&a; // 不合法 |
3、指针可以在定义的时候不初始化,引用必须在定义的时候初始化
Plain Text int *p; // 合法 int &r; // 不合法 int a = 996; int &r = a; // 合法 |
4、指针可以指向NULL,引用不可以为NULL
Plain Text int *p = NULL; // 合法 int &r = NULL; // 不合法 |
5、指针初始化之后可以再改变,引用不可以
Plain Text int a = 996; int *p = &a; // 初始化, p 是 a 的地址 int &r = a; // 初始化, r 是 a 的引用 int b = 885; p = &b; // 合法, p 更改为 b 的地址 r = b; // 不合法, r 不可以再变 |
6、sizeof 的运算结果不同
Plain Text int a = 996; int p = &a; int &r = a; cout << sizeof(p); // 返回 int 类型的大小 cout << sizeof(r); // 返回 int 类型的大小 |
在64位机器上,int* 类型的大小为8个字节,int类型的大小为4个字节。
sizeof 是C/C++ 中的一个操作符(operator),其作用就是返回一个对象或者类型所占的内存字节数。
7、自增运算意义不同
如下图所示,p++之后指向a后面的内存,r++相当于a++。
8、指针和引用作为函数参数时,指针需要检查是否为空,引用不需要
Plain Text void fun_p(int *p) { // 需要检查P是否为空 if (p == NULL) { // do something } } void fun_r(int &r) { // 不需要检查r // do something } |
使用引用的意义和作用:
既然引用时变量的别名,那别名的意义有时什么呢?
引用作为函数的参数,其作用与指针作为函数参数相似,函数对形参的操作,等于对实参本身进行操作;
函数调用时,系统采用
值传递
的方式将实参的值传递给形参,系统会在内存中开辟空间来存储形参变量,并将实参的值复制给形参, 而采用引用作为函数形参,只要传给函数实参的别名,不需要耗费时间在内存中开辟空间存储形参,使用引用,可以提高函数的时间效率,并节省内存空间。
C++
中推荐使用引用而非指针作为函数的参数,指针作为函数形参变量时,调用函数时仍需要在内存中分配空间。
C++
的数组类型是带有长度信息的,引用传递时 如果是数组必须指定数组的长度。
Plain Text #include <iostream> using namespace std; // 函数声明 void swap(int& x, int& y); int main () { // 局部变量声明 int a = 100; int b = 200; cout << "交换前,a 的值:" << a << endl; cout << "交换前,b 的值:" << b << endl; /* 调用函数来交换值 / swap(a, b); cout << "交换后,a 的值:" << a << endl; cout << "交换后,b 的值:" << b << endl; return 0; } // 函数定义 void swap(int& x, int& y) { int temp; temp = x; / 保存地址 x 的值 / x = y; / 把 y 赋值给 x / y = temp; / 把 x 赋值给 y */ return; } 当上面的代码被编译和执行时,它会产生下列结果: 交换前,a 的值: 100 交换前,b 的值: 200 交换后,a 的值: 200 交换后,b 的值: 100 |
指针的意义和作用:
指针是一个变量,其值为另一个变量的地址,即,内存位置的直接地址。就像其他变量或常量一样,您必须在使用指针存储其他变量地址之前,对其进行声明。
暂时就这么多吧!