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

文章介绍了如何使用位运算来解决寻找数组中只出现一次的数字的问题。由于数组内元素出现三次或一次,异或运算无法直接应用。通过计算每个二进制位的和并除以3的余数,可以找到只出现一次的数字的二进制表示。代码示例展示了具体的实现过程,解释了1u的用法,以避免32位整数左移31位时的溢出问题。
摘要由CSDN通过智能技术生成

只出现一次的数字——II

题目链接

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-oM8IiQN6-1690772120702)(C:\Users\HUASHUO\AppData\Roaming\Typora\typora-user-images\image-20230731095705675.png)]

注:本题的解法建立在位运算之上,如果对位运算不太了解,建议先看看👉位运算详解


思路

可能有小伙伴做了只出现一次的数字——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,题目会报错:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-AO3Roro0-1690772120703)(C:\Users\HUASHUO\AppData\Roaming\Typora\typora-user-images\image-20230731104952205.png)]

1 x 31 位的左移不能用类型“int”表示

而之所以不能表示,是因为:对于32位int类型,需要符号位(最高位)保留一个位置。在32位int类型中,左移31位时,会将所有的位都移动到符号位,导致整数溢出。溢出后,结果可能不再是期望的值,而是一个负数或者一个非常大的正数。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Forward♞

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值