懒得看可以直接跳到总结
- *p++与*(p++)是等价的,该表达式返回*p,即p表示的地址所存的数据,并且指针p自增1
- (*p)++ 该表达式返回*p,即p表示的地址所存的数据,然后该数据自增1
- C编译器认为*和++是同优先级操作符,且都是从右至左结合的,所以*p++中的++只作用在p上,和*(p++)意思一样;在(*p)++中,由于()的优先级比*和++都高,所以++作用在()内的表达式*p上。
详解如下
#include <stdio.h>
int main(){
int a[] = {1, 2, 3, 4, 5};
int *p = a;
printf("%d\n", (*p)++); //该语句在屏幕显示 1 实际上该语句可以理解为printf("%d\n", a[0]++); p还是指向a[0]
printf("%d\n", *p);//该语句输出2,此时p指向a[0],但在上一条语句a[0]++在返回一个a[0]的值给printf后就自增为2了
printf("a[0]的地址是%p, p指向了地址%p\n", &a[0], p);//验证p指向了a[0],是a[0]由1自增到了2
a[0] = 1;//将数组还原成a[] = {1, 2, 3, 4, 5};
printf("%d\n", *p++);//输出1,p指向a[1]
printf("%d\n", *p); //该语句输出2,实际上输出的是a[1]
printf("a[1]的地址是%p, p指向了地址%p\n", &a[1], p);//验证p指向了a[1],是p自增了
/*
*由以上可知,*p++可以用来遍历数组
* */
int i;
p = a;//将指针p重新指向 a[0]
for(i=0; i<5; i++)
printf("%d ", *p++);
printf("\n");
return 0;
}
总结
*p++与*(p++)等价,返回*p, 然后指针p++
*p++与*(p++)等价。由于*和++优先级相同,且右结合,所以推测C语言编译器是这样解析处理*p++的:
用一个临时变量接收p,如temp=p;—>p++;return *temp。不过我没有查证我的推测是否正确。可以参考C++对于自增自减运算符的重载,就是这么做的,可以百度“C++ 自增运算符重载”,重载的实现,大概就是这么个套路。总之,实际上是p先增加,然后返回的值,是p增加之前的地址上的值- (*p)++, 返回*p,然后数据*p自增1
- 推理可得,++(*p)、++*p先把数据自增1,然后返回该数据
- 推理可得,*++p先指针p自增,然后返回自增后的p的*p值,和1.所述顺序相反