学过编程语言的童鞋们都知道指针是C语言的精髓,学好了指针就等于学好了C语言,它能够直接对物理地址进行访问,具有双重功能,是嵌入式设计中必不可少的一门语言。C语言功能强大的主要原因就是具有指针结构。指针是一种特殊的数据类型,直接指向目标的存储地址,实现直接访问对象存储空间功能。
指针到底是什么
计算机的内存被划分为多个存储单元,这些存储单元可以以单个或者顺序相连组成一个更大的方式被使用。每一个单独的存储单元都是一个字节,它通常由8个位(bit)组成,每一个位可以表示的值只有0或1.每一个存储单元都被一个及其分配的标识唯一地表示,而这个标识就是地址(指针)。
指针的定义
在C语言中,定义一个普通的变量(如正整数),我们这样做:int i;而定义一个指针变量(指针)我们需要这样做:int *p; 当我们做出int i;这样一个定义时,编译器就会分配一个地址并和i关联起来,而int将限定编译器把这个区域中的内容作为整型看待。现在我们又有了int *p这个定义,假设p是指向变量i的,p中存的是变量i的地址。*表示p是一个指针,而int表示p中所存在的地址对应的变量(即变量i)的类型是int.由此指针定义的一般形式如下:
基类型 *指针名;
指针变量
指针变量:存放内存地址的变量。
指针变量要先初始化后才能够使用,不然就会成为野指针,导致程序崩溃。
#include <stdio.h>
int main(int argc, const char * argv[]) {
int num = 10; //定义一个变量
int *p; //定义一个指针变量
p = # //将变量num的地址赋值给指针变量p;
*p = 30; //为p所指向地址的变量赋值
printf("%p,-- %p; %d == %d", &num, p, num, *p);
return 0;
}
运行结果
0x7fff5fbff75c,-- 0x7fff5fbff75c; 30 == 30Program ended with exit code: 0
以下是错误的赋值:
#include <stdio.h>
int main(int argc, const char * argv[]) {
int num = 10; //定义一个变量
int *p; //定义一个指针变量
//p = # //将变量num的地址赋值给指针变量p;
*p = 30; //为p所指向地址的变量赋值
printf("%p,-- %p; %d == %d", &num, p, num, *p);
return 0;
}
以上代码指针变量未初始化就使用会出现野指针异常,导致程序崩溃。
野指针:没有指向任何地址的指针。(不合法)
空指针:指向地址为null的指针。(合法)
不可以使用一个整型常量赋值给一个指针,因为内存是操作系统分配给我们的,不是我们随便取的。如下:
#include <stdio.h>
int main(int argc, const char * argv[]) {
int *p = 200;
printf("%d", *p);
return 0;
}
运行结果
程序崩溃,出现野指针异常
二级指针
指向指针的指针。定义方式:int **pp; 代码示例如下:
#include <stdio.h>
void printString(char **str)
{
printf("%p\n",str);
*str = "dqk";
}
int main(int argc, const char * argv[]) {
int num = 100; //定义一个变量
int *p; //定义一个指针变量
p = # //为指针变量赋值
int **pp; //定义一个二级指针变量
pp = &p; //为二级指针赋值
**pp = 10; //为二级指针所指向的内存空间赋值
printf("num: %d\n", num);
return 0;
}
运行结果:
num: 10
Program ended with exit code: 0
指针为什么要区分类型
- 变量的地址是变量所在栈存储空间的首地址
- 指针变量仅仅可以存储一个地址编号,如果没有类型,当通过指针就不知道要访问多少个字节的存储空间
- 指针区分类型是为了在通过指针访问它所指向的存储空间的时候能够正确访问
- 如果通过一个char类型的指针操作一个int的变量,如果值的二进制数据超过1字节,那么就造成数据错误
- 如果通过一个int类型的指针操作一个char变量,那么你就会修改了你不该修改的内存,造成程序逻辑错误。