找单身狗--寻找数组中仅出现一次的元素

目录

题目:

思路:

具体实现:

最终代码:


题目:

一个数组中只有两个数字是出现一次,其他所有数字都出现了两次。

编写一个函数找出这两个只出现一次的数字

思路:

我们首先考虑暴力求解 使用遍历 但是遍历的复杂度应该是n^2  

因为其他所有的数字都出现了两次 这时我们考虑或许可以用异或

针对这个题,我们可以先写出两个单身狗的异或和 

然后我们就可以知道哪一位为1 再根据其中一位为1 对所有数组进行分类

两个单身狗肯定在两个不同的类别中 再分别求两个类的异或和 就是两个单身狗

具体实现:

写出基本框架:

void find_single_dog(int arr[],int sz)
{

}
int main()
{
  int arr[]={1,2,3,4,5,1,2,3,4,6};
  int sz= sizeof(arr)/sizeof(arr[0]);
  find_single_dog();
  return 0;
}

两个函数的异或和:

int i = 0;
int ret =0 ;//创建临时变量 承载异或和
for( i=0;i<sz;i++)
{
   ret ^= arr[i];
}

求出异或和中哪一位为1

int pos = 0; //创建临时变量 承载那一位
for(i=0;i<32;i++)
{
  if(((ret>>i)&1)==1) 
     {
      pos = i;
      break; //退出循环, 一位即可
     }
}

根据具体那一位为1 分类  进而分别求出两个单身狗

for (i = 0; i < sz; i++)
{
	if (((arr[i] >> pos) & 1) == 0)
	{
		single1 ^= arr[i];
	}
}
single1 = ret ^single2;

 进而输出就好了 。 

最终代码:

void findTwoNum(int arr[], int n, int * pnum1, int * pnum2)
{
 int i;
 int sum = 0;
​
 for (i = 0; i < 9; i++)
 {
  sum ^= arr[i];
 } //先找到两个数互相异或的结果
​
 int pos;
 for (i = 0; i < 32; i++)
 {
  if (sum & 1 << i)
  {
   pos = i;
   break;
  }
 } //再找到有分歧的一位。在这一位上,两个数一定是一个1一个0
​
 *pnum1 = *pnum2 = 0;
 for (i = 0; i < 10; i++)
 {
  if (arr[i] & 1 << pos)
  {
   *pnum1 ^= arr[i]; //这一位是1的,放在数1里
  }
  else
  {
   *pnum2 ^= arr[i]; //这一位是0的,放在数2里
  }
 }
}

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值