每日一题——只出现一次的数字(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位时,会将所有的位都移动到符号位,导致整数溢出。溢出后,结果可能不再是期望的值,而是一个负数或者一个非常大的正数。

相关推荐
小尧嵌入式5 分钟前
C++11线程库的使用(上)
c语言·开发语言·c++·qt·算法
mywyu_wmb22 分钟前
编译CEF 132.3.3
c语言
wadesir32 分钟前
Judy数组:C语言中的高性能动态数组(全面入门Judy库使用指南)
c语言·开发语言
sheeta199833 分钟前
LeetCode 每日一题笔记 日期:2025.12.14 题目:2147.分隔长廊的方案数
linux·笔记·leetcode
一个平凡而乐于分享的小比特35 分钟前
C语言中 volatile 关键字总结
c语言·volatile
发疯幼稚鬼35 分钟前
插入排序与冒泡排序
c语言·数据结构·算法·排序算法
一个平凡而乐于分享的小比特40 分钟前
static 关键字详解
c语言·static
一个平凡而乐于分享的小比特43 分钟前
C语言内存布局
c语言·const·内存布局
Bigan(安)3 小时前
【奶茶Beta专项】【LVGL9.4源码分析】09-core-global全局核心管理
linux·c语言·mcu·arm·unix
前端小白在前进3 小时前
⭐力扣刷题:螺旋矩阵
算法·leetcode·矩阵