剑指 Offer 56 - II. 数组中数字出现的次数 II

该博客介绍了一种使用位运算的方法来解决数组中只有一个数字出现一次,其他数字出现三次的问题。通过统计每个数字二进制表示的每一位出现的次数,可以找出只出现一次的数字。算法的时间复杂度为O(n),空间复杂度为O(1)。此外,还提到了位运算的其他应用,如非运算和DFS有限自动状态机。
摘要由CSDN通过智能技术生成

题目描述

在一个数组 nums 中除一个数字只出现一次之外,其他数字都出现了三次。请找出那个只出现一次的数字。
示例 1:
输入:nums = [3,4,3,3]
输出:4
示例 2:
输入:nums = [9,1,7,9,7,9,7]
输出:1

位运算 + 逐位统计

在这里插入图片描述
如果一个数字出现三次,那么它的二进制表示的每一位(0或者1)也出现三次。如果把所有出现三次的数字的二进制表示的每一位都分别加起来,那么每一位的和都能被3整除。如果某一位的和能被3整除,那么那个只出现一次的数字二进制表示中对应的那一位是0;否则就是1;

上述思路同样适用于数组中一个数字出现一次,其他数字出现奇数次问题(如果是偶数次,直接用异或 + 分组 就可)。

这种解法的时间效率是O(n)。我们需要一个长度为32的辅助数组存储二进制表示的每一位的和。由于数组的长度是固定的,因此空间效率是O(1)。

代码实现

class Solution {
    public int singleNumber(int[] nums) {//本算法同样适用于数组nums中存在负数的情况
        int[] counts = new int[32];//java int类型有32位,其中首位为符号位
        for(int num : nums) {
            int keyBit = 1;
            //因为(num & keyBit)是二进制的第32位,所以j从31开始自减。
            for(int j = 31; j >= 0; j--) {
                if((num & keyBit)!=0) counts[j]++;
                //用上面的代替这个,因为&操作的答案并不是0或者1,就比如110 & 010,答案是2,并不是1。
                //counts[j] += num & keyBit;
				//这里同样可以通过num的无符号右移>>>来实现,否则带符号右移(>>)左侧会补符号位,对于负数会出错。
            	//但是不推荐这样做,最好不要修改原数组nums的数据
                keyBit = keyBit << 1;//左移没有无符号、带符号的区别,都是在右侧补0
            }
        }
        int res = 0, m = 3;
        for(int i = 31; i >= 0; i--) {
            res += (counts[i] % m) * Math.pow(2,31 - i);
        }
        return res;
    }
}

DFS有限自动状态机

请参考以下文章

作者:Krahets
https://leetcode-cn.com/problems/shu-zu-zhong-shu-zi-chu-xian-de-ci-shu-ii-lcof/solution/mian-shi-ti-56-ii-shu-zu-zhong-shu-zi-chu-xian-d-4/

附,二进制运算符~

非运算:~
10    01
~1 = 0       ~ 0 = 1
~1001= 0110
~按位取反 
5二进制00000101,取反11111010,代表-6 
所以~5-6
byte b = 1;
System.out.println(~b);//输出-2 ~运算会将符号位0变为1,1变为0.
System.out.println(b);//输出1.
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值