[Leetcode]只出现一次的数字

题目

给定一个非空整数数组,除了某个元素只出现一次以外,其余每个元素均出现两次。找出那个只出现了一次的元素。

说明:

你的算法应该具有线性时间复杂度。 你可以不使用额外空间来实现吗?

示例 1:

输入: [2,2,1]
输出: 1

示例 2:

输入: [4,1,2,1,2]
输出: 4

分析

题目中的重点要求:

  1、线性时间复杂度:要求我们的代码时间复杂度最高为O(n),不能有嵌套循环等。

  2、不使用额外空间:要求空间复杂度最高为O(1)。

除此之外,还有重要的信息:

  • 除了某个元素只出现一次以外,其余每个元素均出现两次。

这个条件非常关键,一开始自己审题不清楚没注意到均出现两次这个关键点,按照其他元素出现多次的情况处理了,这样导致思路受限很多。

解法

方法一(比较法):

11 ms40.1 MBjava
// 思路:先对数组进行排序,然后对 nums[i] 和 nums[i + 1]进行比较,如相等,i+=2,继续下一组比较,
// 直到取到不相等的一组。注意:首先这个数组的长度肯定是奇数(目标数字只出现一次,其他所有数字出现
// 两次),所以如果上述步骤没有找到不相等的一组数,那么肯定是数组的最后一个数字是单独出现的。
class Solution {
    public static int singleNumber(int[] nums) {
        Arrays.sort(nums);  // 排序数组
        for (int i = 0; i < nums.length - 1; i += 2) {
            // 找到不相等的一组,直接返回
            if (nums[i] != nums[i + 1]) {
                return nums[i];
            }
        }
        // 如果没有找到不相等的一组数据,直接返回数组的最后一个数字
        return nums[nums.length - 1];
    }
}

方法二(求差法):

68 ms14.9 MBpython3
#思路:先对数组排序,显而易见的,单独出现一次的数据必然是出现在数组下标为偶数的位置(下标从0开始),
#那么所有奇数下标的元素之和减去偶数下标的元素之和,就是需要求得的结果。
class Solution:
    def singleNumber(self, nums: List[int]) -> int:
        nums.sort()
        return sum(nums[0::2])-sum(nums[1::2])

 方法三(集合法)

56 ms14.9 MBpython3
#思路:根据set不重复的特点,将所有元素取出求和再乘二,最后减去原始列表,运算结果就是题目的答案。
class Solution:
    def singleNumber(self, nums: List[int]) -> int:
        return 2*sum(set(nums))-sum(nums)

方法四(异或法)

56 ms14.7 MBpython3
#思路:根据异或运算的特点,相同的数字经过异或运算后结果为0,除单独出现一次的数字外,
#其他数字都是出现两次的,那么这些数字经过异或运算后结果一定是0。而任何数字与0异或都是本身。
#异或也满足交换率,所以对数组所有元素进行异或运算,运算结果就是题目的答案。
class Solution:
    def singleNumber(self, nums: List[int]) -> int:
        num = 0
        for n in nums:
            num = num^n
        return num

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值