【C/C++】运算符问题

问题1:

下面程序的结果是多少?

#include <iostream>
using namespace std; 
int main() 
{
	unsigned char a = 0xA5; 
	unsigned char b = ~a >> 4 + 1;
	// cout << b; 
	printf("b=%d\n", b);
	return 0; 
}

问题2:

用一个表达式,判断一个数X是否是2的N次方(2,4,8,16……),不可用循环语句。

问题3:

下面代码:

int f(int x, int y) 
{
	return (x&y) + ((x^y) >> 1); 
}

(729,271)=_______

问题4:

利用位运算实现两个整数的加法运算,请用代码实现。

========================================================================================

问题1解析

这道题目考查两个知识点:一是类型转换问题;二是运算符的优先级问题。

对于第一个问题:unsigned char b = ~a>>4,在计算这个表达式的时候,编译器会把a和4的值转换为int类型(即所谓整数提升)后再进行计算,当计算结果出来后,再把结果转换成unsigned char 赋值给b。

对于第二个问题:因为“~”的的优先级高于“>>”和“+”,本题的过程是这样的:先对于1010 0101取反0101 1010;再右移,这里有一个问题,是先右移4位再加1呢,还是直接右移5位(4+1)位?因为“+”的优先级高于“>>”,所以直接右移5位。结果是0000 0010.

最后的结果应该是2才对,但把如上的指令放到vs2008中运行,答案居然是250。

那么到底是什么地方出了问题?在调试的过程中进行汇编指令。可以看到高级语句转换为汇编语言以后,是先执行取反再位移的。我们看到eax是16位寄存器,于是在机器中0xA5的寄存器中表达式为0000 0000 1010 0101,取反是1111 1111 0101 1010,那么右移5位是0000 0111 1111 1010,由于是unsigned char型的只能表示低8位的数值,即250。

答案:C

问题2解析

2、4、8、16这样的数转化为二进制是10、100、1000、10000。如果X减1与X做与运算,答案若是0,则X是2的N次方。

答案:!(X&(X-1))

问题3解析

这道题如果使用笨办法来求解,就都转化为二进制然后按位与。但这样的做法显然不是面试官所期待的。仔细观察一下题目,x&y是取相同的位与,这个的结果是x和y相同位和的一半,x^y是取x和y的不同位,结果相当于取了x和y不同位的和,右移相当于除以2,所以这个函数功能是取两个数的平均值。(729+271)/2=500。

答案:500

问题4解析

对于二进制加法,若不考虑进位,则1+1=0,1+0=1,0+1=1,0+0=0,通过对比异或,不难发现,此方法与异或类似。因而排除进位,加法可以用异或实现。

然后考虑进位,0+0的进位是0,1+0的进位为0,只有1+1进的进位才为1,该操作与位运算&操作相似。

那么加法运算可以这样实现:

1)不考虑进位,按位计算各位累加(用异或实现),得值a;

2)然后计算进位,并将进位的值左移,得值b,若b为0,则a就是加法运算的结果,若b不为0,则a+b即得结果(递归调用该函数)。

int Add(int a, int b) 
{
	if (b == 0) return a; 
	int sum = a^b; 
	int carry = (a&b) << 1; 
	return Add(sum, carry); 
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值