鱼C论坛

 找回密码
 立即注册
查看: 868|回复: 12

[已解决]C语言++,--的相关问题

[复制链接]
发表于 2020-11-11 16:05:05 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
#include <stdio.h>
int main()
{
        int i = 8;
        printf("%d\n%d\n%d\n%d\n%d\n%d\n%d\n",i, ++i, --i, i++,i--,-i++,-i--);
}
这里的输出8,8,8,7,8,-7,-8。这个听小甲鱼说的是全部算完再依次赋值,我想的是++i是先输出在计算,i++是先算在输出

所以我的逻辑,i=8; ++i输出的是8,但是算完是9,到--i应该是9了;i++的结果是9,i--是8;-i++是-7,-i--是-8;
有没有大神解释一下;中间咋就不一样了。这个逻辑没明白
最佳答案
2020-11-12 20:44:20
这里有五个程序以,你都运行一下,可以加强你对加加减减运算的理解
#if(0)
#include <stdio.h>
void main()
{
        int i=3;
        int j=4;
        int a = i++;
        int b = ++j;
        printf("%d, %d, %d, %d\n", a, i, b, j);        //运行结果:3, 4, 5, 5
}
/***********************************************************************
分析原因:
a = i++;由于是先执行赋值运算,再自增,所以结果是a=3,i=4;
b = ++j;则因先自增,然后再赋值,所以b,j均为5.
***********************************************************************/

#endif


#if(0)
#include <stdio.h>
void main()
{
        int i=3;
        int j=4;
        int a = i++ + i++;
        int b = ++j + ++j;
        printf("%d, %d, %d, %d\n", a, i, b, j);        //运行结果:6, 5, 12, 6
}
/***********************************************************************
分析原因:
i++的理解应该是执行完整个表达式的其他操作后,然后才自增,所以例子中的a=3+3=6;
而后i再自增2次,i=5;
相反,++j是先自增2次,j=6;然后再参加其它运算,所以b=6+6=12.
***********************************************************************/

#endif


#if(0)
#include <stdio.h>
void main()
{
        int i=3;
        int j=4;
        int a = i++ + i++ + i++;
        int b = ++j + ++j + ++j;
        printf("%d, %d, %d, %d\n", a, i, b, j);        //运行结果:9, 6, 19, 7
}
/***********************************************************************
分析原因:
对于a = i++ + i++ + i++;我们已经没有疑问了,++后置就是执行完整个表达式的其他操作后,
然后才自增,上例中也得到了验证,但 b = ++j + ++j + ++j;又该如何理解呢?

原理表达式中除了预算法本身的优先级外,还有一个结合性问题。
在++j + ++j + ++j;中,因为存在两个同级的+运算,
根据+运算符的左结合性,在编译时,其实是先处理前面的(++j + ++j)这部分,
然后再将此结果再和++j相加。b=(6+6)+7注意最后一个++j是自增了3次的
具体过程参见汇编代码:
        int b = ++j + ++j + ++j;
  0040B7DD mov ecx,dword ptr [ebp-8]
  0040B7E0 add ecx,1
  0040B7E3 mov dword ptr [ebp-8],ecx // 第一个++j
  0040B7E6 mov edx,dword ptr [ebp-8]
  0040B7E9 add edx,1
  0040B7EC mov dword ptr [ebp-8],edx // 第二个++j
  0040B7EF mov eax,dword ptr [ebp-8]
  0040B7F2 add eax,dword ptr [ebp-8] // ++j + ++j
  0040B7F5 mov ecx,dword ptr [ebp-8]
  0040B7F8 add ecx,1
  0040B7FB mov dword ptr [ebp-8],ecx // 第三个++j
  0040B7FE add eax,dword ptr [ebp-8] // ++j + ++j + ++j
  0040B801 mov dword ptr [ebp-10h],eax // 赋值给b

  int a = i++ + i++ + i++;
  0040B7B6 mov eax,dword ptr [ebp-4]
  0040B7B9 add eax,dword ptr [ebp-4] // i+i
  0040B7BC add eax,dword ptr [ebp-4] // i+i+i
  0040B7BF mov dword ptr [ebp-0Ch],eax // 赋值给a
  0040B7C2 mov ecx,dword ptr [ebp-4]
  0040B7C5 add ecx,1
  0040B7C8 mov dword ptr [ebp-4],ecx // 第一次i++
  0040B7CB mov edx,dword ptr [ebp-4]
  0040B7CE add edx,1
  0040B7D1 mov dword ptr [ebp-4],edx // 第二次i++
  0040B7D4 mov eax,dword ptr [ebp-4]
  0040B7D7 add eax,1
  0040B7DA mov dword ptr [ebp-4],eax // 第三次i++
  果然不出所料。到此,++运算符前置后置的问题应该彻底解决了。
***********************************************************************/

#endif


#if(0)
#include <stdio.h>
void main()
{
        int i=1;
        int j=1;
        int a = i++ + i++ + i++ + i++ + i++ + i++ + i++; // 七个
        int b = ++j + ++j + ++j + ++j + ++j + ++j + ++j;
        printf("%d, %d\n", a, b);        //7, 36
        printf("%d, %d\n", i, j);        //8, 8
}
/***********************************************************************
分析原因:
  a = 1+1+1+1+1+1+1 = 7,  i=8
  b = 3+3+4+5+6+7+8 = 36, j=8
***********************************************************************/

#endif


#if(1)
#include<stdio.h>

int main()

{
      int j = 3,i = 5;
      printf("j = 3,  i = 5\n");
          printf("\n");
          j = j * (i++);
          printf("j = j * (i++) 运算后的结果是:%d, %d",j,i);
          printf("       // 先运算,再自加");
          printf("\n");
          j = 3,i = 5;
      printf("\n");
          j = j * (i--);
          printf("j = j * (i--) 运算后的结果是:%d, %d",j,i);
          printf("       // 先运算,再自减");
          printf("\n");
          j = 3,i = 5;
      printf("\n");
      j = j * (++i);
          printf("j = j * (++i) 运算后的结果是:%d, %d",j,i);
          printf("       // 先自加,再运算");
          printf("\n");
          j = 3,i = 5;
      printf("\n");
      j = j * (--i);
          printf("j = j * (--i) 运算后的结果是:%d, %d",j,i);
          printf("       // 先自减,再运算");
          printf("\n");
          j = 3,i = 5;
      printf("\n");
      j = j * (+i);
          printf("j = j * (+i)  运算后的结果是:%d, %d",j,i);
      printf("       // 保持符号再运算");
          printf("\n");
          j = 3,i = 5;
      printf("\n");
      j = j * (-i);
          printf("j = j * (-i)  运算后的结果是:%d, %d",j,i);
          printf("      // 改变符号再运算\n");
          printf("\n");

          printf("单一的 i+ 或单一的 i-                    // 没有意义,并且系统会报错\n");
          printf("\n");

}

#endif

本帖被以下淘专辑推荐:

想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2020-11-12 00:58:40 | 显示全部楼层
这个问题出现过多次了,看这篇帖子里我的答案

https://fishc.com.cn/forum.php?m ... 919&pid=4343099
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-11-12 20:44:20 | 显示全部楼层    本楼为最佳答案   
这里有五个程序以,你都运行一下,可以加强你对加加减减运算的理解
#if(0)
#include <stdio.h>
void main()
{
        int i=3;
        int j=4;
        int a = i++;
        int b = ++j;
        printf("%d, %d, %d, %d\n", a, i, b, j);        //运行结果:3, 4, 5, 5
}
/***********************************************************************
分析原因:
a = i++;由于是先执行赋值运算,再自增,所以结果是a=3,i=4;
b = ++j;则因先自增,然后再赋值,所以b,j均为5.
***********************************************************************/

#endif


#if(0)
#include <stdio.h>
void main()
{
        int i=3;
        int j=4;
        int a = i++ + i++;
        int b = ++j + ++j;
        printf("%d, %d, %d, %d\n", a, i, b, j);        //运行结果:6, 5, 12, 6
}
/***********************************************************************
分析原因:
i++的理解应该是执行完整个表达式的其他操作后,然后才自增,所以例子中的a=3+3=6;
而后i再自增2次,i=5;
相反,++j是先自增2次,j=6;然后再参加其它运算,所以b=6+6=12.
***********************************************************************/

#endif


#if(0)
#include <stdio.h>
void main()
{
        int i=3;
        int j=4;
        int a = i++ + i++ + i++;
        int b = ++j + ++j + ++j;
        printf("%d, %d, %d, %d\n", a, i, b, j);        //运行结果:9, 6, 19, 7
}
/***********************************************************************
分析原因:
对于a = i++ + i++ + i++;我们已经没有疑问了,++后置就是执行完整个表达式的其他操作后,
然后才自增,上例中也得到了验证,但 b = ++j + ++j + ++j;又该如何理解呢?

原理表达式中除了预算法本身的优先级外,还有一个结合性问题。
在++j + ++j + ++j;中,因为存在两个同级的+运算,
根据+运算符的左结合性,在编译时,其实是先处理前面的(++j + ++j)这部分,
然后再将此结果再和++j相加。b=(6+6)+7注意最后一个++j是自增了3次的
具体过程参见汇编代码:
        int b = ++j + ++j + ++j;
  0040B7DD mov ecx,dword ptr [ebp-8]
  0040B7E0 add ecx,1
  0040B7E3 mov dword ptr [ebp-8],ecx // 第一个++j
  0040B7E6 mov edx,dword ptr [ebp-8]
  0040B7E9 add edx,1
  0040B7EC mov dword ptr [ebp-8],edx // 第二个++j
  0040B7EF mov eax,dword ptr [ebp-8]
  0040B7F2 add eax,dword ptr [ebp-8] // ++j + ++j
  0040B7F5 mov ecx,dword ptr [ebp-8]
  0040B7F8 add ecx,1
  0040B7FB mov dword ptr [ebp-8],ecx // 第三个++j
  0040B7FE add eax,dword ptr [ebp-8] // ++j + ++j + ++j
  0040B801 mov dword ptr [ebp-10h],eax // 赋值给b

  int a = i++ + i++ + i++;
  0040B7B6 mov eax,dword ptr [ebp-4]
  0040B7B9 add eax,dword ptr [ebp-4] // i+i
  0040B7BC add eax,dword ptr [ebp-4] // i+i+i
  0040B7BF mov dword ptr [ebp-0Ch],eax // 赋值给a
  0040B7C2 mov ecx,dword ptr [ebp-4]
  0040B7C5 add ecx,1
  0040B7C8 mov dword ptr [ebp-4],ecx // 第一次i++
  0040B7CB mov edx,dword ptr [ebp-4]
  0040B7CE add edx,1
  0040B7D1 mov dword ptr [ebp-4],edx // 第二次i++
  0040B7D4 mov eax,dword ptr [ebp-4]
  0040B7D7 add eax,1
  0040B7DA mov dword ptr [ebp-4],eax // 第三次i++
  果然不出所料。到此,++运算符前置后置的问题应该彻底解决了。
***********************************************************************/

#endif


#if(0)
#include <stdio.h>
void main()
{
        int i=1;
        int j=1;
        int a = i++ + i++ + i++ + i++ + i++ + i++ + i++; // 七个
        int b = ++j + ++j + ++j + ++j + ++j + ++j + ++j;
        printf("%d, %d\n", a, b);        //7, 36
        printf("%d, %d\n", i, j);        //8, 8
}
/***********************************************************************
分析原因:
  a = 1+1+1+1+1+1+1 = 7,  i=8
  b = 3+3+4+5+6+7+8 = 36, j=8
***********************************************************************/

#endif


#if(1)
#include<stdio.h>

int main()

{
      int j = 3,i = 5;
      printf("j = 3,  i = 5\n");
          printf("\n");
          j = j * (i++);
          printf("j = j * (i++) 运算后的结果是:%d, %d",j,i);
          printf("       // 先运算,再自加");
          printf("\n");
          j = 3,i = 5;
      printf("\n");
          j = j * (i--);
          printf("j = j * (i--) 运算后的结果是:%d, %d",j,i);
          printf("       // 先运算,再自减");
          printf("\n");
          j = 3,i = 5;
      printf("\n");
      j = j * (++i);
          printf("j = j * (++i) 运算后的结果是:%d, %d",j,i);
          printf("       // 先自加,再运算");
          printf("\n");
          j = 3,i = 5;
      printf("\n");
      j = j * (--i);
          printf("j = j * (--i) 运算后的结果是:%d, %d",j,i);
          printf("       // 先自减,再运算");
          printf("\n");
          j = 3,i = 5;
      printf("\n");
      j = j * (+i);
          printf("j = j * (+i)  运算后的结果是:%d, %d",j,i);
      printf("       // 保持符号再运算");
          printf("\n");
          j = 3,i = 5;
      printf("\n");
      j = j * (-i);
          printf("j = j * (-i)  运算后的结果是:%d, %d",j,i);
          printf("      // 改变符号再运算\n");
          printf("\n");

          printf("单一的 i+ 或单一的 i-                    // 没有意义,并且系统会报错\n");
          printf("\n");

}

#endif
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2020-11-13 10:02:00 | 显示全部楼层
SHRS23 发表于 2020-11-12 00:58
这个问题出现过多次了,看这篇帖子里我的答案

https://fishc.com.cn/forum.php?mod=redirect&goto=findp ...

学到了~谢谢大佬
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2020-11-13 10:02:33 | 显示全部楼层
风过无痕1989 发表于 2020-11-12 20:44
这里有五个程序以,你都运行一下,可以加强你对加加减减运算的理解

一下子就清晰了很多,拿小本本记下了,谢谢大佬
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-11-13 10:37:00 | 显示全部楼层
小狼爱编程 发表于 2020-11-13 10:02
一下子就清晰了很多,拿小本本记下了,谢谢大佬

客气了,看过你的回复,在你面前,我是初学,可以说还没入门
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-11-13 11:39:26 | 显示全部楼层
++i是先计算再输出,i++是先输出再计算
比如
i=2;
i++;
直接输出i是3
如果是
i=2;
i++;
得到的结果是i=2
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2020-11-13 17:02:32 | 显示全部楼层
风过无痕1989 发表于 2020-11-13 10:37
客气了,看过你的回复,在你面前,我是初学,可以说还没入门

那完了,我连入门都没有,比你还初学
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2020-11-13 17:03:05 | 显示全部楼层
葛明佳 发表于 2020-11-13 11:39
++i是先计算再输出,i++是先输出再计算
比如
直接输出i是3

这两个我给记反了,脑壳痛~
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-11-13 17:12:38 | 显示全部楼层
小狼爱编程 发表于 2020-11-13 17:03
这两个我给记反了,脑壳痛~

++i , 加号在前面,就是先自加,后运算;
i++ , 加号在后面,就是先运算,后自加;
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-11-14 16:12:24 | 显示全部楼层
小狼爱编程 发表于 2020-11-13 17:03
这两个我给记反了,脑壳痛~

哪个在前面就先计算哪个
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2020-11-17 09:32:18 | 显示全部楼层
葛明佳 发表于 2020-11-14 16:12
哪个在前面就先计算哪个

OK谢谢
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2020-11-17 09:32:54 | 显示全部楼层
风过无痕1989 发表于 2020-11-13 17:12
++i , 加号在前面,就是先自加,后运算;
i++ , 加号在后面,就是先运算,后自加;

嗯嗯
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-1-12 13:30

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表