您的位置:首页 > 其它

优先级,结合性,序列点, 你理解清楚了么?

2011-03-25 17:59 267 查看
经常有人问类似i=i++, c=d=c++, a=i++ * i++等表达式的值,每次还都能看到不少错误的回答。 理解这些问题需要弄清楚c语言的表达式求值(evaluation)过程。 今天就抛砖引玉了,希望能终结月经贴。
一个表达式可以分两部分来看:
副作用 - side effect, 我觉得效果这个词比较形象一点。 意思是一个表达式带来的影响
结果 - result , 就是表达式的值

比如i++, 副作用是i+1,结果是i本身。

然后是三个名词:
优先级 - precedence
结合性 - associativity
序列点 - sequence point

很多人错误的理解了优先级和结合性的意思,认为它们决定了一个表达式中各子表达式的求值顺序,这是错误理解开头所提问题的根源。 而实际上:

1. 优先级和结合性是的唯一作用是决定哪些操作数是属于哪些操作符。

2. 优先级和结合性不能表明哪些表达式先求值哪些后求值。
3. 序列点就是程序执行过程中的特定点,子表达式的求值顺序只被序列点决定。
4. C语言只保证,到一个序列点的时候,这个序列点前面的所有side effect都已经完成,而且这个序列点后面的所有side effect都还没有发生。
5. 上一个序列点后到这一个序列点之间,各子表达式的求值顺序未定义。
6. 序列点包括:
a> 函数调用(在参数处理完成,实际进入函数之前)
b> 在操作符逻辑与(&&), 逻辑或(||),条件(?),逗号(, 函数参数列表里不算)的第一个操作数之后。
c> 一个完整的表达式后面,比如分号后面或者if里面的控制表达式等。
7. 对于序列点,一个重要的原则是: 在上一个序列点后到这一个序列点之间最多改变一个对象的值一次。
例如表达式:

a=i + i++ * i++ - b;
由于++的优先级最高,*次之,+和-的优先级一样,但结合性是从左到右,所以这个表达式会被解析成:

a=(i + ((i++) * (i++))) - b;
这就是优先级和结合性的唯一作用,只是表明操作数应该往哪个操作符上靠。 至于其中的子表达式比如两个i++什么时候发生,什么顺序发生,都是未定义的,不确定的。

而像下面这个:

i++ && i++ && i++
其执行顺序就是确定的,因为&&是序列点,每到一个&&, 就可以保证其前面所有的side effect已经发生。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: