1.指针是什么?
指针理解的2个要点:
- 指针是内存中一个最小单元的编号,也就是地址
- 平时口语中说的指针,通常指的是指针变量,是用来存放内存地址的变量
总结:指针就是地址,口语中说的指针通常指的是指针变量
指针是用来存放地址的,地址是唯一标示一块地址空间的。
指针的大小在32位平台是4个字节,在64位平台是8个字节。
2.指针和指针类型
1.指针的类型决定了,对指针解引用的时候有多大的权限(能操作几个字节)。比如: char* 的指针解引用就只能访问一个字节,而 int* 的指针的解引用就能访问四个字节。
2.指针的类型决定了指针加1减1操作时,跳过几个字节(决定了指针的步长)。
野指针
概念: 野指针就是指针指向的位置是不可知的(随机的、不正确的、没有明确限制的)。
1.指针未初始化
#include <stdio.h>
int main()
{
int *p;//局部变量指针未初始化,默认为随机值
*p = 20;//非法访问内存,这里的p就是野指针
return 0;
}
p没有初始化,就意味着没有明确的指向;一个局部变量不初始化,放的是随机值。
2.指针越界访问
#include <stdio.h>
int main()
{
int arr[10] = {0};
int *p = arr;
int i = 0;
for(i=0; i<=11; i++)
{
//当指针指向的范围超出数组arr的范围时,p就是野指针
*(p++) = i;
}
return 0;
}
3.指针指向的空间释放
int* test()
{
int a = 10;
return &a;
}//程序退出该函数后,该空间的内容被释放
int main()
{
int* p = test();//p仍然记得a的地址,但里面的内容已经被释放了,后面p再被使用就是野指针
return 0;
}
4.如何规避野指针
- 指针初始化
- 小心指针越界
- 指针指向空间释放即使置NULL
- 避免返回局部变量的地址
- 指针使用之前检查有效性
指针运算
1.指针±整数
#define N_VALUES 5
float values[N_VALUES];
float *vp;
//指针+-整数;指针的关系运算
for (vp = &values[0]; vp < &values[N_VALUES];)
{
*vp++ = 0;//可以拆成*vp = 0;vp++;
}
int main()
{
int arr[10] = {0};
int i = 0;
int* p = arr;
for(i = 0;i<10;i++)
{
*(p+i) = i;
}
return 0;
}
2.指针-指针
指针-指针的绝对值得到的是指针和指针之间的元素个数。
不是所有的指针都能相减,指向同一块空间的两个指针才能相减。
int my_strlen(char *s)
{
char *p = s;
while(*p != '\0' )
p++;
return p-s;
}
3.指针的关系运算
for(vp = &values[N_VALUES]; vp > &values[0];)
{
*--vp = 0;
}
代码简化, 这将代码修改如下:
for(vp = &values[N_VALUES-1]; vp >= &values[0];vp--)
{
*vp = 0;
}
实际在绝大部分的编译器上是可以顺利完成任务的,然而我们还是应该避免这样写,因为标准并不保证它可行。
标准规定:
允许指向数组元素的指针与指向数组最后一个元素后面的那个内存位置的指针比较,但是不允许与指向第一个元素之前的那个内存位置的指针进行比较。+
二级指针
int main()
{
int a = 10;
int* pa = &a;//pa是一个指针变量,一级指针变量
int** ppa = &pa;//ppa是一个二级指针变量,二级指针变量是用来存放一级指针变量的地址的。
}
指针数组
存放指针的数组就是指针数组。
int main()
{
int* parr[10] = {&a,&b,&c};
}