关于指针,其是C语言的重点,C语言学的好坏,其实就是指针学的好坏。其实指针并不复杂,学习指针,要正确的理解指针。
指针是一种数据类型
指针也是一种变量,占有内存空间,用来保存内存地址
指针就是告诉编译器,开辟4个字节的存储空间(32位系统),无论是几级指针都是一样的
*p操作内存
在指针声明时,*
号表示所声明的变量为指针
在指针使用时,*
号表示操作指针所指向的内存空间中的值*p
相当于通过地址(p变量的值)找到一块内存;然后操作内存*p
放在等号的左边赋值(给内存赋值)*p
放在等号的右边取值(从内存获取值)
指针变量和它指向的内存块是两个不同的概念
- 给p赋值
p = 0x1111;
只会改变指针变量值,不会改变所指的内容;p = p +1;`
p++` - 给
*p
赋值*p='a';
不会改变指针变量的值,只会改变所指的内存块的值 - 等号左边
*p
表示给内存赋值,等号右边*p
表示取值,含义完全不同 - 等号左边
char *p
(处理字符串,指针指向字符串首地址:char *p = buf;p++;
) - 保证所指的内存块能修改,即不要出现野指针和指向系统内存区域以及修改常量区数据
指针是一种数据类型,是指它指向的内存空间的数据类型
指针步长(p++),根据所致内存空间的数据类型来确定p++ = (unsigned char )p + sizeof(a);
结论:指针的步长,根据所指内存空间类型来定
通过*p
/*p++
来改变变量的值是指针存在的最大意义
指针变量和它指向的内存块变量
这完全是两码事
指针指向某个变量,就是把某个变量地址否给指针
指针实际上记录的就是地址
*p
间接赋值成立条件
需要满足三个条件
- 2个变量(通常一个实参,一个形参)
- 建立关系,实参取地址赋给形参指针
*p
形参去间接修改实参的值1
2
3
4
5int num = 0; //实参
int *p = NULL;
p = #
num = 1;
*p = 2; //通过*形参 间接 地改变实参的值
间接赋值的应用场景
在一个函数之内:
1
*p1++ = *p2++;
函数调用:
1
2
3
4int getStr(int *a)
{
···
}
函数调用时,用n指针(形参)改变n-1指针(实参)的值
改变0级指针(e.g. int num = 1
)的值有2种方式
改变1级指针(e.g. char *p = 0x1111
)的值,有2种方式
改变2级指针的(e.g. char **pp = 0x1111
)的值,有2种方式
函数调用时,形参传给实参,用实参取地址,传给形参,在被调用函数里面用*p
,来改变实参,把运算结果传出来
间接赋值的推论 ==> 指针做函数参数
指针作为函数参数的精髓
- 用1级指针(形参)去改变了0级指针(实参)的值(通过
*p
去间接修改了实参的在值) - 用2级指针(形参)去改变了1级指针(实参)的值(通过
*p
去间接修改了实参的在值) - 用3级指针(形参)去改变了2级指针(实参)的值(通过
*p
去间接修改了实参的在值) - 用n级指针(形参)去改变了n-1级指针(实参)的值(通过
*p
去间接修改了实参的在值)
深入理解指针必须和内存四区概念相结合,注意指针的输入输出特性
####主调函数 被调函数
- 主调函数可把堆区、栈区、全局数据内存地址传给被调用函数
- 被调用函数只能返回堆区、全局数据
内存分配方式
指针做函数参数,是有输入和输出特性的
应用指针必须和函数调用相结合(指针做函数参数)
序号 | 指针 | 堆栈 | 主调函数参数 | 被调函数参数 | 备注 |
---|---|---|---|---|---|
1 | 1级指针(做输入) | 堆 | 分配 | 使用 | 一般禁用 |
栈 | 分配 | 使用 | 常用 | ||
2 | 1级指针(做输出) | 栈 | 使用 | 传出结果 | 常用 |
3 | 2级指针(做输入) | 堆 | 分配 | 使用 | 一般禁用 |
栈 | 分配 | 使用 | 常用 | ||
4 | 2级指针(做输出) | 堆 | 使用 | 分配 | 常用,不建议用,一般转为2 |
5 | 3级指针(做输出) | 堆 | 使用 | 分配 | 不常用 |
1级指针做输入1
2int showbuf(char *p);
int showArray(int *p; int num);
1级指针做输出1
int getLen(char *pFileName, int *pFileLen);
2级指针做输入1
2int main(int argc, char *args[]);//指针数组
int shouMatrix(int [3][4], int len);//二维数组字符串
2级指针做输出1
2
3int getData(char **data, int *dataLen);
int getData_Free(void *data);
int getData_Free(void **data);//避免野指针
3级指针做输出1
2int getFileAllLine(char ***content, int *pLine);
int getFileAllLine_Free(cahr ***content, int *pLine);