C语言:寻找两只单身狗

 在一个数组中只有两个数字出现一次,其余数字都是成对出现,编写函数找出这两个数字。

示例:输入:1 2 3 4 3 2 1 5

           输出:4 5

方法一、暴力求解

 设置一个标识符tmp为1,利用两个指针遍历判断整个数组,用continue跳过自身和自身的判断,如果数组后面有和自身相等的,则将标识符改为0,最后利用标识符作为判断条件打印单身狗数字。

void Find_signal(int* p, int sz)
{
	int i = 0;
	int* flag = p;
	int j = 0;
	for (j = 0; j < sz; j++)
	{
		int tmp = 1;
		for (i = 0; i < sz; i++)
		{
			if (i == j)
			{
				continue;
			}
			if (*(p + j) == *(flag + i))
			{
				tmp = 0;
				break;
			}
		}
		if (tmp==1)
		{
			printf("%d\n", *(p + j));
		}
	}
}

总结:暴力求解方法的问题在于它并不是获得两个数字,而是直接打印出来,对于只想获取到数字的某些情况来说此方法并不适用。

方法二、异或

我们知道按(二进制)位异或操作符(^),相同为0,相异为1。所以两个相同的数字异或是0,且0与任何一个数字异或还是那个数字。

所以首先我们可以对数组中的所有元素进行异或,这样实际上就是数组中两个不同的元素进行异或,如1,2,3,6,1,2,3,5进行异或如下

 接着将整个数组的元素分成两组,将两个单身狗数字分别放在两组之中。那么对数组该怎么分组呢,以什么条件进行分组呢。

以两个数字二进制的第n位不同作为区分,如上图5和6的二进制的倒数第二位不同,所以可以将数组中元素二进制的倒数第二位是0的放在一组中(1,1,3,3,5),是1的放在一组中(2,2,6)。

接着对两组数分别进行全部异或,这样就找到了两个数字。


void Find_signal(int arr[], int sz, int* dog1, int* dog2)
{
	int i = 0;
	int ret = 0;
	//异或
	for (i = 0; i < sz; i++)
	{
		ret = ret ^ arr[i];
	}
	//计算ret的二进制中左右边的第几位1
	int pos = 0;
	for (pos = 0; pos < 32; pos++)
	{
		if (((ret >> pos) & 1) == 1)
		{
			break;
		}
	}
	//分组
	for (i = 0; i < sz; i++)
	{
		if (((arr[i] >> pos) & 1) == 1)
		{
			*dog1 ^= arr[i];
		}
		else
		{
			*dog2 ^= arr[i];
		}
	}

}

总结: 这种方法比较好,可以将两个单身狗数字通过指针赋值过去,直接获得。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值