编程之美 1.5 快速找出故障机器

题意:

题目:假设一个机器只存储一个标号为ID的记录,假设每份数据保存2个备份,这样就有2个机器存储了相同的数据。其中ID是小于10亿的整数
问题1、在某个时间,如果得到一个数据文件ID的列表。
是否能够快速的找到这个表中仅出现一次的ID?即快速找出出现故障的机器存储的数据ID。
问题2、如果有两台机器死机呢?(假设同一个数据的俩个备份不会同时丢失,
即列表中缺少的是两个不等的ID)
扩展题、如果所有的机子都有三个备份,也就是说同一ID的机子有三台。
而且同时又有三台机子死机,还能用上面的方法解决吗?
如果有N台备份,又同时有N台机器死机呢?
相关问题、给你一副杂乱的扑克牌(不包括大小王),任意从其中抽出一张牌,
怎样用最简单的方法来知道抽出的是1~13中的那一张?(不要求知道花色)

解法一:

对于列表中的每一个ID,遍历列表查找是否存在相同的ID,利用数组记录每个ID出现的次数,
遍历完,次数小于2的就是要找的;
时间复杂度为O(N),空间复杂度为O(N)。空间复杂度在N很大时不理想。

解法二:

利用hash表。遍历列表,对于每一个ID,先检查hash表中是否有与之相同的ID,
若有,则从Hash表中删除该ID;否则,将该ID加入到hash表中。
这样,遍历完列表后,hash表中剩下的那一个元素即为所求ID。
时间复杂度为O(N),空间复杂度在最好的情况下为O(1),在最坏的情况下为O(N)。

解法三:

 利用异或运算,因为所以ID出现2次,只有一个出现一次,而a^a=0,
所以将这个列表中的所有ID异或后的值即为所求ID。
时间复杂度为O(N),空间复杂度为O(1)。在时间和空间上,基本已经达到最优。

 若是有2个ID仅出现一次,设为A和B,那么所有ID的异或值为A^B,无法确定A与B的值;
但是仍然可以找到,若A=B,A^B=0,则可以通过求和的方法A=B=(所有ID和-所以正常工作ID和)/2
若A!=B,A^B!=0,那么这个值最高为1,显然A与B当中有且只有一个数的相同位为1;

所以根据这一位,我们可以将a和b分开,并将数分成两组。
我们在结果数字中找到第一个为1的位的位置,记为第N位。
现在我们以第N位是不是1为标准把原数组中的数字分成两个子数组,
第一个子数组中每个数字的第N位都为1,第二个子数组的每个数字的第N位都为0。
所以把第N位为1的异或 结果num1, 把第N位为1的异或 结果num1,

#include<iostream>
using namespace std;

void FindTwoDifferentNum(int a[],int n,int &num1,int &num2)
//返回数组a中两个只出现一次的数字num1和num2
{
	int i,ans=0,k;

	for(i=0;i<n;i++)//先全部异或,其结果是这两个只出现一次的数字的异或
		ans^=a[i];
	
	num1=ans;//保存ans 
	
	k=0;//k表示的是第N位 a和b的不同 
	while((ans&1)==0)
	{
		ans>>=1;
		k++;
	}

	ans=num1;
	for(i=0;i<n;i++)//第n位不同 将全为1的异或 得到num1 
	{
		if((a[i]>>count)&1)
			num1^=a[i];
	}
	num2=num1^ans;//num1和全部异或的结果异或,就是num2
}

int main()
{
	int a[]={1,5,3,4,2,3,1,5,4,6};
	int num1, num2;
	FindTwoDifferentNum(a,sizeof(a)/sizeof(int),num1,num2);
	cout<<num1<<"  "<<num2<<endl;
}

解法四:

利用“不变量”。所有ID的和为一个不变量,对剩下ID求和。
所有ID的和与剩下ID的和之差即为所求ID。
由于所有ID之和可以事先算好,所以,该方法也可以在O(N)时间,O(1)空间内解决。

问题2,用2个方程解决
首先计算出初始未丢失之前,所有ID之和。  
然后计算出丢失之后的ID之和,然后(1)(2)结果进行相减操作,得到方程x+ y = a。  
利用丢失前后平方和之差,来与(2)进行联立,得到方程x*x+y*y=b。  
对两方程进行联立,即可以求出最终的结果。 

五、扩展问题 

如果同时有A个备份,同时有B台机器发生故障,怎么解决?

         在B小于4的情况下,还可以考虑采用解法四构造方程解决。但是,B超过4的情 况下,方程势必很复杂,建议采用解法三,当然,解法三需要改进。需要对存入Hash表的ID计数,当计数值达到A时再将这个ID从hash表中删除。

六、相关问题

给你一副杂乱的扑克牌(不包括大小王),任意从其中抽出一张牌,

怎样用最简单的方法来知道抽出的是1~13中的那一张?(不要求知道花色)?

         直接采用解法四;首先算好所有牌的和(1+…+13)x4=364,然后分别减去留下的牌点数,就得到抽出的是哪一张。

    原文作者:小人物_cipher
    原文地址: https://blog.csdn.net/u012605629/article/details/41323805
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞