只出现一次的数字——II
注:本题的解法建立在位运算之上,如果对位运算不太了解,建议先看看👉位运算详解
思路
可能有小伙伴做了只出现一次的数字——I后认为这题也可以用异或运算来解决,但是我们需要注意到,本题中数组中的数据,要么出现了三次,要么只出现了一次,我们用异或运算不能做到消除出现了三次的数据而只留下出现了一次的数据,因此需要换一种方法。
这一题,我们可以通过计算二进制位来解决问题。
如果不看只出现了一次的数据,那么由于每个元素都恰出现了三次,那么每个二进制位要么是3个0,要么是3个1,而加上这多出的一个数据,就会使某些二进制位变成4个1,而二进制位仍为3个1的就说明只出现一次的元素的这个二进制位为0。
可以得出结论:
所求数据的第
i
个二进制位就是数组中所有元素的第i
个二进制位之和除以3
的余数。
举个例子,对于数组[2,2,2,3]
这样一来,我们就可以通过操作(nums[i] >> i) & 1
来求得每个数据的第i
个二进制位,相加得到total
,再用total % 3
得到只出现一次的数字的第i
个二进制位,如果这一位为1,最后再利用ret |= 1u << i
就可以的到最后的结果
实现代码
int singleNumber(int* nums, int numsSize){ int ret = 0; //设置返回值 for(int i = 0; i < 32; i++) { int total = 0; //计算第i位二进制的和 for(int j = 0; j < numsSize; j++) total += (nums[j] >> i) & 1; //如果和取模为1,那么就说明返回值第i位的二进制位也为1 if(total % 3) ret |= 1u << i; } return ret; }
可能有小伙伴会疑惑1u
是什么意思,为什么要写1u
而不是1
1u
的意思是将1
这个int型数据转换为无符号数,如果我们用1
,题目会报错:
1 x 31 位的左移不能用类型“int”表示
而之所以不能表示,是因为:对于32位int类型,需要符号位(最高位)保留一个位置。在32位int类型中,左移31位时,会将所有的位都移动到符号位,导致整数溢出。溢出后,结果可能不再是期望的值,而是一个负数或者一个非常大的正数。