今天更新一篇小短文,希望对你有帮助。
int func(int temp){return!!temp;}
不知道你是否看过上面类似的代码,两个感叹号出现在代码中,难道代码也有思想,也需要表达情感吗?
刚学习 C语言的时候,你应该经常看到过 1 个感叹号的情况,比如:
if(one!=two){.......}-----------------------------------------if(!temp){......}-----------------------------------typedef enum {RESET = 0, SET = !RESET} FlagStatus, ITStatus;-----------------------------------------one=!temp;
但两个感叹号估计就很难见到了。
鱼鹰大学四年、工作三年都没见过这种写法,直到前段时间看 Linux 源码,才接触到,第一次看到时非常惊讶,怎么还有这种写法?
为什么要用两个感叹号,作用是什么,只是为了表现 C 语言的奇技淫巧吗?
仔细想过后才惊叹其中的巧妙。
假设一个字节变量 byte,可代表范围 0~255,0 代表其中一种含义,1 ~255 代表另一种含义(你可能会问,怎么不直接用 0 和 1 表示,因为这个变量本身不只有 0 和 1,只是在另一个使用的地方才会只使用二值含义,总之会有这种情况)。
如果我要用另一个变量 bit 来表示这两种含义,一般情况我们会这么做:
int func(unsigned char byte){ unsigned char bit; if(byte==0){bit=0;}else{ bit = 1;}return bit;}
更优雅简单一点是这样写:
int func(unsigned char byte){bit=byte?1:0; return bit;}
但不管哪一个,都不如第一个简单高效。
简单可以很容易看出来,高效何在?
它不需要判断语句(判断语句在单片机中可能影响不是很大,但在有多级缓存的情况下,影响可能很大,这就是为什么 linux 中用 likely() 之类的进行优化)。
这样,不管原先的byte 是什么值,都将变成 0 或1。
这样一来,如果调用者使用如下方式:
if(func() == 1){}或者 if(func()){ }
都不会出现问题。
对于负数也是如此,只要是为了把 0 单独分开,都可以采用这种方式。
这在底层开发中也非常实用。
比如 GPIO 有个引脚号需要判断是 0 或 1,一般这样:
bit=(GPIOB->IDR&GPIO_Pin_4) >> 4;
或者bit=(GPIOB->IDR&GPIO_Pin_4)? 1 : 0;
上一种确实也是不错的选择,但是这里需要修改两个地方,修改时很容易遗忘,所以不如下面这种简单:
bit=!!(GPIOB->IDR&GPIO_Pin_4);
如果换个 IO ,需要修改代码时,只要修改一次就搞定,相当方便,所以建议大家使用上面那种方式获取位的值。
而从汇编的角度来看,两次 ! 也只需要一条指令搞定:
效率不输移位方式!
本文转载自鱼鹰谈单片机
审核编辑:汤梓红
-
Linux
+关注
关注
87文章
11292浏览量
209326 -
C语言
+关注
关注
180文章
7604浏览量
136686 -
源码
+关注
关注
8文章
639浏览量
29185 -
代码
+关注
关注
30文章
4779浏览量
68521
原文标题:7 年了,没见过代码中出现过两个感叹号
文章出处:【微信号:玩点嵌入式,微信公众号:玩点嵌入式】欢迎添加关注!文章转载请注明出处。
发布评论请先 登录
相关推荐
评论