HDU4149 Magic Potion 异或运算(菜鸟杯)

Problem Address:http://acm.hdu.edu.cn/showproblem.php?pid=4149


【前言】


当时比赛的时候没有做出坑爹的第六题。

第七题心甘情愿不会做,解题报告的那句话不怎么看懂。

不过后来赶着做大作业去了,一直没搞。

今天稍有点轻松,于是动手解决。

琢磨着就想出来了。确实还算可以的一道题。只是自己想不到。


【思路】


1007. Magic Potion(位运算):
二进制位,从最低位往上考虑,一位一位的做,因为有8个x,如果有奇数个为1,那么一个个的异或之后个数还是奇数个,但是所有加起来再异或却为偶数了,改变了~~~~注意进位。



以上是解题报告的原话。

确实要一位一位的做。

先看第一位。

设未异或的为x,异或后用xm表示。

假若8个xm中有奇数个为1,无论m的这一位是什么,原来的8个x肯定有奇数个位1。则相加结果肯定为1。

如果相加结果异或完也为1,说明m的第一位肯定是0。若异或完为0,说明m为1。

若m为1,由于xm中有奇数个为1,说明8个x中为0的个数是xm为1的个数。

若m为0,由于xm中有奇数个1,说明原来x中为1的个数是xm为1的个数。

同样的,若xm偶数个位1,则x也偶数个为1,则相加结果为0,然后跟上面做相同的判断即可。

关键的地方是进位。

从一开始需要增加一个标志位p,初始化为0。

假设m为1,xm中有奇数个1,那么做完以上计算之后要把xm为1的个数加到p上,同时在下一次运算的时候将p右移。

由于增加了标志位,那么判断相加结果的时候就不只是原来的那个数,还应该考虑当前p的最低位。


【代码】


#include <iostream>
using namespace std;

int x[10];

int main()
{
	int m, p;
	int i;
	int t;
	scanf("%d", &t);
	while(t--)
	{
		for (i=1; i<=9; i++) scanf("%d", &x[i]);
		m = 0;
		p = 0;
		int k=0;
		while(1)
		{
			for (i=1; i<=8; i++)
			{
				if (x[i]!=0) break;
			}
			p >>= 1;
			if (i>8 && p==0 && x[9]==0) break;
			int ct = 0;
			for (i=1; i<=8; i++)
			{
				if (x[i]&1) ct++;
			}
			int y = (ct&1);
			int z = ((x[9]&1)+(p&1))&1;
			int tm = y^z;
			if (tm)
			{
				p += (8-ct);
			}
			else
			{
				p += ct;
			}
			m += (tm<<k);
			k++;
			for (i=1; i<=9; i++) x[i]>>=1;
		}
		printf("%d\n", m);
	}
	return 0;
}


琢磨着就想出来了。确实还算可以的一道题。只是自己想不
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值