C语言指针
1.&运算符
&是取地址符号;
#include<stdio.h>
int main()
{
int a=20;
printf("a 的值为%d\n",a);
printf("a 的地址为%d\n",&a);
return 0;
}
//打印结果:
a 的值为20
a 的地址为1703740(1703740为a地址0019FF3C的十进制)
& 符号的作用很清晰, &a就是取变量a的地址。
2. *运算符
*是定义指针变量,即指向内存单元的指针
#include<stdio.h>
int main()
{
int a=20;
int *ip;
ip=a;
printf("ip 的值为%d\n",ip);
printf("*ip 的值为%d\n",*ip);
return 0;
}
//打印结果:
ip 的值为20
应该会编译出错,ip=a语法错误(可以试一试*ip=a编译没错,但是输出会卡死)。*表示一个指针变量,指向内存单元,指向内存单元,应该就是地址吧,那么ip变量应该被赋值一个地址,前边说的 & 不就表示一个地址吗,试一下:
#include <stdio.h>
int main ()
{
int a=5;
int *b;
b=&a;
printf("a 的值为 %d\n",a);
printf("&a 的值为 %d\n",&a);
printf("b的值为%d\n",b);
printf("*b 的值为 %d\n",*b);
}
//打印结果:
a 的值为 5
&a 的值为 1703740
b的值为1703740
*b 的值为 5
定义变量时int* a;int* 表示取指针,在地址前加*表示这个地址的内容。
3.函数指针和指针函数
函数指针的定义:
returnType (*pointerName)(param list);
指针函数的定义:
returnType *pointerName(param list);
returnType为函数返回值类型,pointerName 为指针名称,param list 为函数参数列表,函数指针赋值为函数地址。注意:( )的优先级高于*,第一个括号不能省略,如果写作returnType *pointerName(paramlist);就成了函数原型(指针函数),它表明函数的返回值类型为returnType *,返回值类型为1个指针。
调用函数直接调用就好了,为什么还要定义一个函数指针呢,才接触时很不理解,后了解到由于函数指针存储的是一个内存地址,在传参的时候传入一个地址比传入一个地址对应的函数名更高效,运算速度更快。
4.数组指针
数组是一种结构类型,在某种意义上来说数组名就是指针,可以转化为指向其实体的指针,指针则是变量,仅仅意味着数组的存放地址,数组名概念比较宽泛,也更加复杂,而指针仅仅代表数组的首地址
#include <stdio.h>
int main()
{
int a[]={1,2,3,4,5};
int *p=a;
printf("数组a首地址%d\n",&a);//数组a首地址1703724
printf("数组a首地址%d\n",p);//数组a首地址1703724
printf("a所占空间%d\n",sizeof(a));//a所占空间20(a代表整个数组,大小为4*5=20)
printf("*p所占内存空间%d\n",sizeof(p));//*p所占内存空间4(*p仅仅指向首地址)
printf("*p指向地址的内容%d\n",*p);//*p指向地址的内容1
return 0;
}
指向指针的指针
#include<stdio.h>
#include<stdlib.h>
int main(){
char *s[]={"aaa","bbb","ccc","ddd"};
char **p;//定义指向指针的指针
int k;
for(k=0;k<4;k++){
p=&s[k];//指针赋值为地址 如果不加则为s[k]的内容,会出错
printf("%s\n",*p);//使用*p访问指针所指向地址的内容
}
return 0;
}
总结
&a表示a变量的地址,scanf("%d",&a)中&a就是将输入的值存入a变量的地址中
*ip定义一个指针变量ip,ip代表指针变量,赋值内容为地址(即指向变量的地址)
*ip表示值是地址的变量,访问的是ip地址内容
在数组中指针变量仅仅代表数组的首地址,数组名就数组的首地址
int p; //这是一个普通的整型变量
int *p; //首先从P 处开始,先与*结合,所以说明P 是一个指针,然后再与int 结合,
说明指针所指向的内容的类型为int 型.所以P是一个返回整型数据的指针
int p[3]; //首先从P 处开始,先与[]结合,说明P 是一个数组,然后与int 结合,
说明数组里的元素是整型的,所以P 是一个由整型数据组成的数组
int *p[3]; //首先从P 处开始,先与[]结合,因为其优先级比*高,所以P 是一个数组,然后再与*结合,
说明数组里的元素是指针类型,然后再与int 结合,说明指针所指向的内容的类型是整型的,
所以P 是一个由返回整型数据的指针所组成的数组
int (*p)[3]; //首先从P 处开始,先与*结合,说明P 是一个指针然后再与[]结合(与"()"这步可以忽略,
只是为了改变优先级),说明指针所指向的内容是一个数组,然后再与int 结合,
说明数组里的元素是整型的.所以P 是一个指向由整型数据组成的数组的指针
int **p; //首先从P 开始,先与*结合,说是P 是一个指针,然后再与*结合,说明指针所指向的元素是指针,
然后再与int 结合,说明该指针所指向的元素是整型数据.
由于二级指针以及更高级的指针极少用在复杂的类型中,
所以后面更复杂的类型我们就不考虑多级指针了,最多只考虑一级指针.
int p(int); //从P 处起,先与()结合,说明P 是一个函数,然后进入()里分析,
说明该函数有一个整型变量的参数,然后再与外面的int 结合,
说明函数的返回值是一个整型数据
Int (*p)(int); //从P 处开始,先与指针结合,说明P 是一个指针,然后与()结合,
说明指针指向的是一个函数,然后再与()里的int 结合,说明函数有一个int 型的参数,
再与最外层的int 结合,说明函数的返回类型是整型,
所以P 是一个指向有一个整型参数且返回类型为整型的函数的指针
int *(*p(int))[3]; //可以先跳过,不看这个类型,过于复杂从P 开始,先与()结合,
说明P 是一个函数,然后进入()里面,与int 结合,说明函数有一个整型变量参数,
然后再与外面的*结合,说明函数返回的是一个指针,,然后到最外面一层,先与[]结合,
说明返回的指针指向的是一个数组,然后再与*结合,说明数组里的元素是指针,
然后再与int 结合,说明指针指向的内容是整型数据.
所以P 是一个参数为一个整数据且返回一个指向由整型指针变量组成的数组的指针变量的函数.