指针简介
一、什么是指针
指针是一种数据类型,可以定义变量,保存的内容是地址
二、作用
指针保存的是地址,通过指针变量可以直接操作地址,转而可以操作寄存器地址,从而实现直接访
问硬件的功能,这是C语言与其他语言最大的不同。
三、大小
既然是指针是一种数据类型,那么他的大小是多少 ~
因为在计算机中 地址是固定的长度(由操作系统的位数决定),所以指针变量的长度也是固定
的,不同系统位数的指针变量的字节长度不同,64位系统地址指针变量长度为8字节,32位系统指
针变量长度为4字节
四、定义指针指向数据的类型的缘由
因为指针的地址属性—步长(做加1操作所移动的字节数),不同类型的指针变量会保存不同步长
的地址 地址+操作数(实际是移动了相应数据类型大小的字节数)=改变后的地址
五、指针的间接访问
变量的内容保存在指针指向的地址里,所以指针通过保存变量的首地址来间接访问变量的内容
六、指针运算符*和&
* 间接运算符 提取指针指向地址的内容
& 取地址符 取出保存数据的地址
*p = a =》p = &a
相当于逆运算,&是内容向外取地址,*是地址向内取内容
七、指针的定义方式
int a;
int* p = &a;
=》
int a;
int* p;
p = &a;
指针阅读——右左法则
int* p
p 没有()[] 所以是一个变量 *表示指针标识符,说明是指针变量保存的是一个地址, int说明是
该指针变量指向一个整型的变量
八、指针的偏移
p++ p中保存的是变量地址,++表示地址加1个步长,指向下一个内容变量
(*p)++ *p是指针里的内容,++ 表示该地址里的内容加1,即相应数据类型的值加上1
多级指针
一、作用
保存上一级指针的地址,应用在函数的传参
注意:
相同类型指针之间才可以赋值
void* 万能指针,用来接受任意类型的值 (不知道返回值的情况下),但不能直接打印内容,步长
为1
指针的加减,可以减不能加——指针+指针(返回error),指针-指针(返回两个数据地址之间的步
长(指针类型相同情况下)
野指针
what(是什么)
指针变量里保存的地址对应空间无访问权限(指针所指向的地址空间无访问权限)
why(为什么出现)
1.定义后未初始化的指针
2、释放结束之后的指针
3、越界访问的指针(指向分配空间之后的指针)
野指针出现产生的问题
内存泄露
运行时错误
内存错误(段错误)
how(怎么避免)
养车良好的编码习惯
1、定义变量时必须初始化
当指针变量作为指向工具时,定义初始化NULL
当指针变量指向的空间赋值时,需要动态申请空间
2、使用时
检查内存空间是否分配成功
初始化内存空间(清零内存)
防止越界访问
3、使用结束
必须释放空间
一定将指针再次置为NULL
NULL是什么
简单理解0,但不等于0,NULL用于指针和对象,0用于数值,因为NULL大多数情况下指向0,所
以#defined NULL 0;但并不是说NULL=0
内存空间分配
1、分配方式
静态方式:开销小,但是空间利用率不高
动态分配:开销大,提高空间利用
malloc
malloc只开辟空间,不进行类型检查,返回一个无类型指针,必须加强制类型转换
指针自身 = (指针类型*)malloc(sizeof(指针类型)*数据数量)
int *p = NULL;
int n = 10;
p = (int *)malloc (sizeof(int) * n);
长度:sizeof(类型)*数量
多次申请空间时系统是如何做到空间的不重复使用?
在使用malloc开辟一段空间之后,系统会在这段空间之前做一个标记(0或1),当malloc函数开辟
空间如果遇到标记为0就在此开辟,如果为1说明此空间正在被使用
calloc
开辟空间时,对空间进行初始化
用法:void*calloc(元素的数目,每个元素的大小)
调用成功后,malloc和calloc都返回分配空间的首地址
realloc
更改已配置空间的内存大小
用法:realloc(更改指针名,更改成的大小)
内存减小时:直接减,如果原来的空间位置内容未被修改,依旧可以打印原来的数
内存增大
当前内存段后有足够的内存空间大小,直接扩展,realloc()将返回原指针
当前内存后的空间不够,就使用堆中第一个满足要求的内存块,将目前数据复制到新的位置,并将
原来的数据块释放掉,放回新的内存块位置
如果申请失败,讲返回NULL,此时,原来指针依旧有效
free函数
释放malloc calloc realloc函数给指针变量分配的内存空间时注意要使用时添加函数库
(#include<stdlib.h>) 要在堆空间内开辟内存