三,使用指针时需要注意的问题:
空指针:值为NULL的指针叫做空指针,如果对空指针进行解引用一定会产生段错误
NULL 一般作为一种错误标志,当一个函数的返回值是指针类型时,可以使用NULL函数执行出错的返回结果
如何避免空指针带来的段错误:
使用来历不明的指针先做判断
if(NULL == p)
1,当函数的参数是指针,别人传给你的指针可能是空指针
2,从函数获取的返回值是指针类型是,可能就是空指针
tap:NULL 在绝大多数系统中是0,个别是1.
野指针:
指向不确定的内存空间的指针叫做野指针
对野指针解引用的后果:
1,一切正常
2,段错误
3,脏数据
野指针比空指针危害更严重,因为它无法判断出来,而且可能是隐藏性的错误,短时间不暴露
所有的野指针都是程序员自己制造出来的,如何避免产生野指针:
1,定义指针变量时一定要初始化
2,函数不要返回栈内存(函数内局部变量)的地址
3,指针指向的内存被释放后,指针变量要及时置空NULL
四,指针的运算
指针变量中存储的是整数,理论上整数可以使用的运算符它都可以用,但是绝大多数运算符都是无意义的
p+n 指针当前地址+指针类型宽度*n 前进n个元素
p-n 指针当前地址-指针类型宽度*n 回退n个元素
p-p1 (指针编号-指针编号)/指针类型宽度 计算两个指针之间间隔了多少个指针元素
五,指针与const
当我们提高传参效率而使用指针作为函数参数时,传参效率提高了,但是变量被共享也有了被修改的风险,可以使用const来保护
const int *p; 保护指针所指向的内存不能被修改
int const *p; 保护指针所指向的内存不能被修改
int* const p; 保护指针变量不能被修改
const int* const p; 保护指针变量和指针指向的内存都不能修改
int const* const p; 同上
六,指针数组和数组指针
指针数组:由指针变量组成的数组,它的成员都是类型相同的指针变量
类型* arr[长度];
数组指针:是专门指向数组的指针
类型 (*arrp)[长度];
七,数组名与指针:
数组名其实就是一种特殊的指针
数组名是一个常量,不能修改它的值,数组名没有自己的存储空间,它与数组首地址是映射关系
数组名 == &数组名
指针变量是是拥有自己的存储空间,它与所指向的内存是指向关系
当指针变量指向数组首地址是,指针可以当作数组名使用,数组名也可以当作指针使用
数组名[i] == *(数组名+i)
*(p+i) == p[i]
当数组作为函数的参数是就蜕变成了指针,所以长度丢失
八,二级指针
指向指针的指针,里面存放的是指针变量的地址
定义: 类型名** 变量名_pp;
赋值: 变量名_pp = &指针变量
解引用:*变量名_pp <==> 指针变量
**变量名_pp <==> *指针变量 <==> 数据
tap 当需要函数之间共享指针变量,传递指针的地址(二级指针)
九,函数指针
函数名就是一个地址,函数名代表了函数在代码段中所处的入口位置
调用函数本质上是让进程跳转到函数所在的代码段中执行二进制指令
如果函数指针中存储了函数的的首地址,对函数指针解引用可以执行函数
函数指针就是指向函数的指针,它里面存储的是函数在代码段中所处的入口位置地址
返回值 (*p)(类型1,类型2,,,.....)
int func(int num1,double num2);
int (*funcp)(int,double);
可以通过函数指针,把函数当作参数传递给另一个函数,这种方式称之为函数回调模式
当函数中需要调用后面写的函数时,可以在函数的参数中提供函数指针,调用后面写的函数,函数回调
浅学C++(3)学习C语言(指针 续)
于 2022-07-18 16:34:38 首次发布