每日一题——只出现一次的数字(II)

只出现一次的数字------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就可以的到最后的结果

实现代码

c 复制代码
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位时,会将所有的位都移动到符号位,导致整数溢出。溢出后,结果可能不再是期望的值,而是一个负数或者一个非常大的正数。

相关推荐
wangjialelele10 小时前
平衡二叉搜索树:AVL树和红黑树
java·c语言·开发语言·数据结构·c++·算法·深度优先
森G11 小时前
七、04ledc-sdk--------makefile有变化
linux·c语言·arm开发·c++·ubuntu
历程里程碑11 小时前
普通数组----合并区间
java·数据结构·python·算法·leetcode·职场和发展·tornado
weixin_3954489111 小时前
mult_yolov5_post_copy.c_cursor_0205
c语言·python·yolo
Z9fish12 小时前
sse哈工大C语言编程练习20
c语言·开发语言·算法
晓131312 小时前
第六章 【C语言篇:结构体&位运算】 结构体、位运算全面解析
c语言·算法
iAkuya12 小时前
(leetcode)力扣100 61分割回文串(回溯,动归)
算法·leetcode·职场和发展
梵刹古音12 小时前
【C语言】 指针与数据结构操作
c语言·数据结构·算法
VT.馒头12 小时前
【力扣】2695. 包装数组
前端·javascript·算法·leetcode·职场和发展·typescript
刘琦沛在进步13 小时前
【C / C++】引用和函数重载的介绍
c语言·开发语言·c++