什么是指针:
指针是一种特殊得数据类型,使用它可以定义指针变量,指针变量中存储得是整型数据,代表了内存得编号,通过这个编号可以访问对应得内存。
为什么要使用指针:
1.函数之间相互独立,但有时候需要共享变量
传参是值传递
全局变量容易命名冲突
使用数组还需要传递长度
函数和函数之间命名空间是相互独立得,但地址空间是同一个,所有指针可以解决这个问题
2.由于函数与函数之间是值传递(拷贝):对于字节数多的变量,值传递的效率比较低,如果传递的是变量地址,只需要传递4|8个字节,提高传参效率
3.由于堆内存无法取名字,它不能像data,bss,stack让变量名于内存建立联系,只能使用指针记录推内存的地址来使用堆内存。
如何使用指针:
定义:类型* 变量名_p; int* num_p;
1.指针变量与普通变量的用法有很大的区别,建议在取名是结尾加个p以示区分。
2.指针的类型表示存储的是什么类型的数据,它决定了通过这个指针可以访问的字节数。
3.一个*只能定义一个指针变量
int* p1,p2,p3; p1是指针变量,p2\p3都是int类型变量
int* p1,* p2, * p3;p1,p2,p3都是指针变量
4.指针变量与普通变量一样都是默认值是随机的,一般初始化为MULL
赋值:
变量名_p = 地址;必须是有权限有意义的内存地址
指向栈内存;
int num = 0;
int* p = &num
指向堆内存:
int* p=malloc(n);
解引用: *p
通过指针变量中记录的内存编号去访问对应的内存,该过程可能出现段错误,原因是里面存储的内存编号是非法地
注意:访问的字节数是由指针变量的类型决定的
使用指针要注意的问题:
空指针: 值为NULL的指针变量的叫空指针,如果进行引用 一定会产生段错误
NULL也是一种错误标准,如果一个函数返回值是指针类型,当函数执行出错时返回值时NULL
如何避免空指针带来的段错误:使用一些来历不明的指针变量前,都做一次判断。
1.当函数的参数是指针,别人传给你的可能是空指针
2.从函数获取的返回值是指针时,也可能是空指针
if (NULL == p) if(!p)
注意:NULL在绝大数操作系统中是0;在个别系统是1
野指针:任何指向不确定的内存的指针
例如 int* p;
解引用野指针可能的后果:
1.段错误
2.脏数据
3.一切正常
野指针比空指针危害更严重,因为它无法判断出来,而且可能是隐藏的错误时间不暴露。
所有的野指针都是程序员字节制造出来的,任何避免产生野指针:
1.定义指针变量时一定要初始化;
2.函数不返回栈内存的地址
3.指针指向的内存释放后,指针变量要及时置空
指针的运算:
指针变量存储的整数,理论上整形数据能使用的运算符它都可以,但是大多数的运算是没有意义的 指针+n <=>宽度*n+指针 前进n个元素
指针-n <=>指针 - 宽度*n 后退n个元素
指针- 指针 <=> (指针-指针)/宽度 计算两个指针之间相隔了但是个元素
const与指针:
const int* p:
当我为了提高传参效率而使用指针时,传参的效率提高了,但是意味着变量有着被修改的风险,这种写法可以保护指针所指的内存。
const int* p;保护指针所指的内存不可修改
int const* p;同上
int* const p;保护指针所指的变量不可修改
const int* const p;保护指针变量,指针变量所指向的内存都不可修改
int const* const p;同上
指针数组与数据指针:
指针数组:
是由指针组成的数组,它的成员全都是指针变量
类型*数组名[长度];
数组指针:
专门指向数组的指针
类型(*数组名)[长度];
指针与数组名:
数组名就是一种特殊的指针,它是常量,不能改变它的值,数组名与数组的内存是一种映射关系,它没有自己的存储空间
数组名 == &数组名
指针变量有自己的存储空间,如果存储的是数组的首地址,指针可以当做数组使用,数组名也可以当指针 数组名[i] == *(数组名+i)
注意:数组作为函数的参数时就蜕变了指针,只有4个字节,长度丢失
二级指针:
二级指针就是指向指针的指针,里面存储的是指针变量的地址
定义: 类型** 变量名_pp;
赋值:变量名_pp = &指针;
解引用:
*变量名_pp <=> 指针;
**变量名_pp <=> *指针;
函数指针:
函数名就是一个地址(整数),它代表了在代码段中函数所处的位置
函数指针是指lei向函数的指针,里面存储是函数在代码段中的位置
返回值(*函数名)(类型1 参数名1,类型2 参数2, …)
int scanf(const shar *format, …);
可以通过函数指针把函数当做参数传递给另一个函数。
void qsort(void *base, size_t nmemb, size_t size,
int( *compar)(const void *,const void *))