Chinaunix首页 | 论坛 | 博客
  • 博客访问: 1131989
  • 博文数量: 300
  • 博客积分: 37
  • 博客等级: 民兵
  • 技术积分: 772
  • 用 户 组: 普通用户
  • 注册时间: 2012-02-26 04:46
文章分类
文章存档

2017年(4)

2016年(7)

2015年(19)

2014年(72)

2013年(71)

2012年(127)

分类: C/C++

2014-04-25 20:22:23

原文地址:细说c语言的优先级 作者:j3kljs02398j

Table
0. 为什么要掌握优先级
1. 优先级
1.1 优先级图表
1.2 运算符实例
1.3 优先级顺口溜
2. 结合性
3. 参考资料

    写代码的时候,常会翻看的一个表就是“c语言运算符优先级表”。c的运算符优先级常常很让人头疼。其实,在大学里学习c的时候,老师告诉大家这个不用一定背下来,用的时候可以找书,或者加小括号就可以了。我听了,但是后来发现错了。很多人都听了,但不是每个人都发现这是错的。以至于有人觉得把优先级背下来是“没事闲的”()。

0. 为什么要掌握优先级

    想想这两个问题:
        a. 读别人的代码,遇到优先级问题看不懂,怎么办?
        b. 一堆的括号,美观吗?
        本想贴一张画来装饰墙壁,却用了一堆纸来固定!
        有人说代码写多了,自然就会了。这个是很宽泛的说法。看你写的代码的水准,有些东西可能你一直都接触不到,何谈熟练。有些东西一定要梳理,总结。

1. 优先级

1.1 优先级图表


  •     优先级最高者不是真正意义上的运算符,包括:数组下标,函数调用,结构体成员选择符。
  •     单目运算符的优先级次之。(! ~ ++ -- - (type) * & sizeof)
  •     然后是双目运算符。双目运算符里, 算数运算符(* / % + -)优先级最高, 移位(<< >>)次之, 关系运算符(< <= > >= != ==)再次之, 接着是位运算符(& ^ | ),逻辑运算符(&& ||) 条件运算符(?: 三目),赋值运算符(= ...)。
  •     任何一个逻辑运算符的优先级低于任何一个关系运算符。
  •     移位运算符的优先级比算数运算符要低,但是比关系运算符要高。

1.2 运算符实例
    a. while (c = getc(in) != EOF)
        putc(c, out)
        循环的意思是复制一个文件到另一个文件。但是由于!=的优先级比赋值运算符的优先级高,所以c被赋予了getc()的返回值与EOF比较后的布尔值,结果向out中写入了一堆1.
    b. 解释下面几个声明
    char *p[];
    char (*p)[];
    int *fp();
    int (*fp)();

    char *p[]
        常常被错误的理解为指向字符数组的指针。
        正确的是p一个数组,里面元素是指向字符的指针类型。
    char (*p)[]
        p是指向指向字符数组的指针。
    int *fp()
        常常错误理解为函数指针,该函数返回int类型。
        正确的是fp是一个函数,他返回一个执行int的指针。
    int (*fp)()
        fp是函数指针,该函数返回int类型。
    
    c. 解释下面的表达式
    *p.f;
    val & mask != 0;
    max = val1 > val2 ? val1 : val2;

    *p.f
        对p去f偏移,作为指针,然后进行解引用。相当与*(p.f),因为.的优先级高与*。比较(*p).f。
    val & mask != 0
        相当与val & (mask != 0).
    max = val1 > val2 ? val1 : val2
        相当与 max = (val1 > val2 ? val1 :val2).
    
    d. 一个复杂的声明

    char *(* c[10])(int **p);
    1. 有两个小括号,小括号的结合行是自左向右,所以我们先关注第一个小括号,简化声明(*c[10])();
        c是一个数组,里面放10指针,后面紧跟这一个括号,所以这些指针是函数指针。
    2. 关注第二个括号,(int **p)
        显然p是函数的参数,它是个指向指针的指针。
    3. 这看最前面的*
        char *说明该函数的返回值是一个指向字符的指针。
    4. 这个声明的意思就是:c是一个数组,里面有10个函数指针,指向的函数返回指针,指向字符,函数的参数是指向int类型指针的指针。

1.3 优先级顺口溜
    醋坛酸味灌
    味落跳福豆

    共44个运算符

    醋-初等,4个: ( ) [ ] -> 指向结构体成员 . 结构体成员
    坛-单目,9个: ! ~ ++ -- -负号 (类型) *指针 &取地址 sizeof长度
    酸-算术,5个: * / % + -减
    味-位移,2个: << >>
    灌-关系,6个: < <= > >= == 等于 != 不等于
    味-位逻,3个: & 按位与 ^ 按位异或 | 按位或
    落-逻辑,2个: && 逻辑与 || 逻辑或
    跳-条件,1个,三目: ? :
    福-赋值,11个: = += -= *= /= %= >>= <<= &= ^= |=
    豆-逗号,1个: ,

    我更推荐在原理上理解以及长时间的使用来掌握,这顺口溜是网上刚刚翻到的,觉得挺好,辅助记忆。

2. 结合性
    在标准C语言的文档里,对操作符的结合性并没有作出非常清楚的解释。一个满分的回答是:它是仲裁者,在几个操作符具有相同的优先级时决定先执行哪一个。
    看例子:
    int a, b = 1, c = 2;
    a = b = c;
    这个表达式只有赋值符,这样优先级就无法帮助我们呢决定哪个操作先执行。如果a = b先执行,然后 b = c执行。那么a最终取1。如果b = c先执行, a = b后执行,那么a最终取2。到底哪一个先执行?看结合性,复制的结合性是右至左,所以b = c,然后a = b。
    同级的操作符,结合性相同。如果在计算表达式的值时候需要考虑结合性,那么最好把这个表达式一分为二。

3. 参考资料
    《c Traps and Pitfalls》 Andrew Koenig著, ISBN 978-7-115-17179-5
    《c 专家编程》 Peter Van Der Linden 著, ISBN 978-7-115-17180-1
    顺口溜 http://blog.sina.com.cn/s/blog_4e64e2290100be0z.html
阅读(977) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~