免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 3751 | 回复: 11
打印 上一主题 下一主题

[C] 下面的代码运行后输出结果为什么都是one one [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2013-10-28 19:24 |只看该作者 |倒序浏览
        static const char *msg[] = {"one","two","three","four","five"};
        char *get(int i)
        {
                static char buf[20];
                strcpy(buf, msg[i]);
                return buf;
        }       
        int main()
        {
                printf("%s %s\n", get(0), get(1));
                return 0;       
        }

论坛徽章:
59
2015年亚洲杯之约旦
日期:2015-01-27 21:27:392015年亚洲杯之日本
日期:2015-02-06 22:09:41拜羊年徽章
日期:2015-03-03 16:15:432015年辞旧岁徽章
日期:2015-03-03 16:54:152015年迎新春徽章
日期:2015-03-04 09:50:282015元宵节徽章
日期:2015-03-06 15:50:392015年亚洲杯之阿联酋
日期:2015-03-19 17:39:302015年亚洲杯之中国
日期:2015-03-23 18:52:23巳蛇
日期:2014-12-14 22:44:03双子座
日期:2014-12-10 21:39:16处女座
日期:2014-12-02 08:03:17天蝎座
日期:2014-07-21 19:08:47
2 [报告]
发表于 2013-10-28 19:32 |只看该作者
static char buf[20];

论坛徽章:
1
丑牛
日期:2013-09-29 19:04:50
3 [报告]
发表于 2013-10-28 22:32 |只看该作者
folklore 发表于 2013-10-28 19:32
static char buf[20];



你的解释是错的。

论坛徽章:
17
处女座
日期:2013-08-27 09:59:352015亚冠之柏太阳神
日期:2015-07-30 10:16:402015亚冠之萨济拖拉机
日期:2015-07-29 18:58:182015年亚洲杯之巴勒斯坦
日期:2015-03-06 17:38:17摩羯座
日期:2014-12-11 21:31:34戌狗
日期:2014-07-20 20:57:32子鼠
日期:2014-05-15 16:25:21亥猪
日期:2014-02-11 17:32:05丑牛
日期:2014-01-20 15:45:51丑牛
日期:2013-10-22 11:12:56双子座
日期:2013-10-18 16:28:17白羊座
日期:2013-10-18 10:50:45
4 [报告]
发表于 2013-10-28 23:17 |只看该作者
回复 3# solu


    2楼标记出来了,由于使用了static(所以这个函数是不可重入的),使得get函数做为实参在传递给printf时变量副作用没有结束,而这将导致行为未定义(在函数调用后并传递实参前有一个序列点),标准规定任何变量在两个序列点之间多于一次修改的结果是未定义的。

论坛徽章:
0
5 [报告]
发表于 2013-10-28 23:25 |只看该作者
  1. #include <stdio.h>
  2. #include <string.h>

  3. static const char *msg[] = {"one","two","three","four","five"};
  4. char *get(int i)
  5. {
  6.         if (1 == i)
  7.         {
  8.                 printf("i=1\n");
  9.         }
  10.         else
  11.         {
  12.                 printf("i=0\n");
  13.         }
  14.         static char buf[20];
  15.         strcpy(buf, msg[i]);
  16.         return buf;
  17. }

  18. int main()
  19. {
  20.         printf("%s %s\n", get(1), get(0));
  21.         return 0;
  22. }
复制代码
输出的结果是:
i=0
i=1
two two
应该是printf()从后向前执行,最终buf指向的内容为 two\0 ,所以输出的都是two

论坛徽章:
1
丑牛
日期:2013-09-29 19:04:50
6 [报告]
发表于 2013-10-28 23:40 |只看该作者
myworkstation 发表于 2013-10-28 23:17
回复 3# solu



未定义行为是什么意思? 以这个例子来说, 是指两个get() 的 输出是一样的吧?

下面这样的话, 为什么输出是0 1.

  1. #include <stdio.h>
  2. #include <string.h>
  3. static const char *msg[] = {"one","two","three","four","five"};
  4. int get(int i)
  5. {
  6.         static int ret;
  7.         ret = i;
  8.         return ret;
  9. }
  10. int main()
  11. {
  12.     printf("%d %d\n", get(0), get(1));
  13.     return 0;
  14. }
复制代码

论坛徽章:
5
双鱼座
日期:2013-11-26 17:56:26狮子座
日期:2013-11-29 15:41:32处女座
日期:2014-02-21 11:59:07技术图书徽章
日期:2014-03-06 15:33:53技术图书徽章
日期:2014-03-06 15:39:30
7 [报告]
发表于 2013-10-28 23:46 |只看该作者
回复 4# myworkstation


    这个跟序列点有关吗?我觉得就是因为printf函数调用参数的先后顺序的关系导致的这个问题。这个printf里面,已经调用了get,2个get之间应该有2个序列点,应该是不影响static变量的赋值阿?

C++2003标准中定义的序列点

   ·完整声明之后

   ·完整表达式之后

   ·进入函数时与退出函数时

   ·|| && ?: , 四个操作符的第一个操作数之后

第三条讲述了get调用前和调用后也算也个序列点的

论坛徽章:
12
巳蛇
日期:2013-09-16 15:32:242015年辞旧岁徽章
日期:2015-03-03 16:54:152015年亚洲杯之约旦
日期:2015-02-11 14:38:37双鱼座
日期:2015-01-05 11:05:47戌狗
日期:2014-12-08 09:41:18戌狗
日期:2014-08-15 09:29:29双子座
日期:2014-08-05 09:17:17卯兔
日期:2014-06-08 15:32:18巳蛇
日期:2014-01-27 08:47:08白羊座
日期:2013-11-28 21:04:15巨蟹座
日期:2013-11-13 21:58:012015年亚洲杯之科威特
日期:2015-04-17 16:51:51
8 [报告]
发表于 2013-10-29 09:18 |只看该作者
2楼4楼解释的很好,某个喷货就不去搭理了。

由于是static的char数组,所以printf后两个参数是同一个地址,所以最后的结果要么是两个one,要么是两个two。两次get调用的顺序是不能保证的,所以行为是未定义的。

论坛徽章:
17
处女座
日期:2013-08-27 09:59:352015亚冠之柏太阳神
日期:2015-07-30 10:16:402015亚冠之萨济拖拉机
日期:2015-07-29 18:58:182015年亚洲杯之巴勒斯坦
日期:2015-03-06 17:38:17摩羯座
日期:2014-12-11 21:31:34戌狗
日期:2014-07-20 20:57:32子鼠
日期:2014-05-15 16:25:21亥猪
日期:2014-02-11 17:32:05丑牛
日期:2014-01-20 15:45:51丑牛
日期:2013-10-22 11:12:56双子座
日期:2013-10-18 16:28:17白羊座
日期:2013-10-18 10:50:45
9 [报告]
发表于 2013-10-29 10:01 |只看该作者
回复 7# bottles


    C标准中有一句话非常明确的指出了这个问题:If an attempt is made to modify the result of a function call or to access it after the next sequence point, the behavior is undefined.

论坛徽章:
17
处女座
日期:2013-08-27 09:59:352015亚冠之柏太阳神
日期:2015-07-30 10:16:402015亚冠之萨济拖拉机
日期:2015-07-29 18:58:182015年亚洲杯之巴勒斯坦
日期:2015-03-06 17:38:17摩羯座
日期:2014-12-11 21:31:34戌狗
日期:2014-07-20 20:57:32子鼠
日期:2014-05-15 16:25:21亥猪
日期:2014-02-11 17:32:05丑牛
日期:2014-01-20 15:45:51丑牛
日期:2013-10-22 11:12:56双子座
日期:2013-10-18 16:28:17白羊座
日期:2013-10-18 10:50:45
10 [报告]
发表于 2013-10-29 10:09 |只看该作者
回复 6# solu


    因为你改成了非指针的标量类型,两个函数返回值不再是相互依赖的(但是计算过程有依赖),尽管如此你的这个代码不一定在所有编译器上运行的结果都相同,另外因为CPU的乱序执行还是有可能产生其它结果。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP