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

本文解析了HDU 4149 Magic Potion题目,通过位运算来解决问题,详细介绍了如何逐位进行异或操作,并考虑进位情况。提供了完整的C++实现代码。

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;
}


琢磨着就想出来了。确实还算可以的一道题。只是自己想不
### 2024 HDU运算 示例 教程 #### 什么是位运算? 位运算是指对二进制数中的每一位进行操作。常见的位运算符包括按位与 (`&`)、按位或 (`|`)、按位异或 (`^`)、左移 (`<<`) 和右移 (`>>`)。 #### 杭州电子科技大学(HDU)位运算题目解析 对于给定的代码片段: ```cpp for (int i = 1; i < (1 << n); i++) { if ((i >> 1) & i) continue; } ``` 这段代码用于遍历 `n` 个元素的所有子集,并过滤掉某些特定条件下的子集。具体解释如下: - `(1 << n)` 表示将数字 `1` 左移 `n` 位,相当于计算 \(2^n\) 的值[^3]。 - `i >> 1` 将变量 `i` 右移一位,相当于除以 2。 - `(i >> 1) & i` 判断当前子集中是否存在相邻位置都为 1 的情况。如果存在,则跳过该次循环。 #### 实际应用案例 假设有一个数组 `[a, b, c]`,即 `n=3`,则上述代码会生成并筛选这些子集: - 子集表示法:`{}` 对应于二进制 `000` - `{a}` 对应于二进制 `001` - `{b}` 对应于二进制 `010` - `{c}` 对应于二进制 `100` - `{a, b}` 对应于二进制 `011` (被过滤) - `{a, c}` 对应于二进制 `101` - `{b, c}` 对应于二进制 `110` (被过滤) - `{a, b, c}` 对应于二进制 `111` (被过滤) 通过这种方式,可以有效地排除那些不符合特定条件的组合。 #### Python 中的位运算示例 以下是类似的逻辑在 Python 中的实现方式: ```python def generate_subsets(n): result = [] for i in range(1, 1 << n): # 遍历所有可能的子集 if not (i & (i - 1)): # 过滤掉含有连续 '1' 的子集 subset = [j for j in range(n) if (i >> j) & 1] result.append(subset) return result print(generate_subsets(3)) ``` 此函数将会输出符合条件的子集列表。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值