C语言中指针和自增运算符结合时的运算顺序问题

本文通过实例详细解析了C语言中指针与++/--运算符的六种组合方式,包括*p++, (*p)++, *(p++), ++*p, ++(*p), *(++p)的运算顺序及特点,帮助读者理解并掌握复杂的指针运算。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

转载请注明出处,否则将追究法律责任http://blog.youkuaiyun.com/xingjiarong/article/details/47071225

在C语言中,当指针运算符和++或者–结合时很容易分不清运算顺序,在这里总结一下,下面一共分析6中组合: * p++,(* p)++,* (p++),++* p,++( * p), * (++p)。

先看段代码以及输出:

#include<stdio.h>
int main()
{
    int a[3]={1,3,5};
    int *p=a;
    printf("----------------1----------------\n");
    printf("%d\n",*p++);
    printf("%d\n",*p);
    int i;
    for(i=0;i<3;i++)
        printf("%d ",a[i]);
    printf("\n");
    printf("----------------2----------------\n");
    p=a;//reset data
    printf("%d\n",(*p)++);
    printf("%d\n",*p);
    for(i=0;i<3;i++)
        printf("%d ",a[i]);
    printf("\n");
    printf("----------------3----------------\n");
    a[0]=1;//reset data
    p=a;
    printf("%d\n",*(p++));
    printf("%d\n",*p);
    for(i=0;i<3;i++)
        printf("%d ",a[i]);
    printf("\n");
    printf("----------------4----------------\n");
    p=a;
    printf("%d\n",++*p);
    printf("%d\n",*p);
    for(i=0;i<3;i++)
        printf("%d ",a[i]);
    printf("\n");
    printf("----------------5----------------\n");
    p=a;
    a[0]=1;
    printf("%d\n",++(*p));
    printf("%d\n",*p);
    for(i=0;i<3;i++)
        printf("%d ",a[i]);
    printf("\n");
    printf("----------------6----------------\n");
    p=a;
    a[0]=1;
    printf("%d\n",*(++p));
    printf("%d\n",*p);
    for(i=0;i<3;i++)
        printf("%d ",a[i]);
    printf("\n");
    return 0;
}

输出结果是这样的:
这里写图片描述

第一组:*p++,它的运算顺序是先返回 *p的值,然后p再++。
第二组:(*p)++,他的运算顺序是先返回 *p的值,然后 *p的值再++,这一点从运算后的数组a的值可以看出来。
第三组:*(p++),运算顺序是先返回 *p的值,然后p再++,也就是说它和 *p++的运算顺序一样。

这三组都是先返回 *p的值,区别就在于到底是p++还是 *p的值++。

第四组:++*p,先将 *p的值++,然后再返回 *p的值。
第五组:++(*p),先将 *p的值++,然后再返回 *p的值,所以它和++ *p是一样的。
第六组: * (++p),先将p的值++,然后再返回 * p的值,和 *++p是等价的。

这三组的特点是最后都是返回 *p的值,不同点在于是 *p先++还是p++。

在 C 语言中,指针运算符的优先级关系是一个重要的概念。根据提供的参考资料[^1],可以得出结论:**指针解引用运算符 (`*`) /自减运算符 (`++`, `--`) 的优先级是相同的**,并且它们的结合方向是从右向左。 这意味着当遇到表达式如 `*p++` 或类似的组合,解析顺序遵循从右向左的原则。例如,在表达式 `*p++` 中,实际的操作顺序如下: 1. 首先执行 `p++`,这表示取当前指针所指向的内容,并准备将指针本身加一个单位。 2. 接着执行 `*` 解引用操作,取得原指针未更新前指向的那个内存单元中的值。 这种行为可以通过实验验证[^2]。如果我们将此逻辑应用于代码片段,则可以看到程序的行为完全符合预期——即先读取旧址处的数据再移动指针位置。 另外值得注意的是关于前置与后置版本的区别。对于像 `(*p)++` 这样的情况,它意味着先取出该地址上的内容并通过加法改变其内部存储值;而如果是 `*++p` ,则代表先进位到下一个元素然后再做访问动作[^3]。 综上所述,在涉及指针及其相关算子的应用过程中,理解清楚各个符号之间相互作用规律至关重要,这样才能写出既安全又高效的源码。 ```c // 示例演示指针与自运算的关系 #include <stdio.h> int main(void){ int numbers[] = {10, 20 ,30}; int *pointer = numbers; printf("Value before increment : %d\n", *pointer); // 输出第一个数 printf("After post-increment : %d\n", *pointer++); // 再次输出同一个数因为post-inc还没生效 printf("Next value after move : %d\n", *pointer); // 移动后的第二个数 pointer--; // 返回第一步状态 (*pointer)++; printf("Modified original data: %d\n", *pointer); return 0; } ```
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值