题目
给定一个非空整数数组,除了某个元素只出现一次以外,其余每个元素均出现两次。找出那个只出现了一次的元素。
说明:
你的算法应该具有线性时间复杂度。 你可以不使用额外空间来实现吗?
示例 1:
输入: [2,2,1] 输出: 1
示例 2:
输入: [4,1,2,1,2] 输出: 4
分析
题目中的重点要求:
1、线性时间复杂度:要求我们的代码时间复杂度最高为O(n),不能有嵌套循环等。
2、不使用额外空间:要求空间复杂度最高为O(1)。
除此之外,还有重要的信息:
- 除了某个元素只出现一次以外,其余每个元素均出现两次。
这个条件非常关键,一开始自己审题不清楚没注意到均出现两次这个关键点,按照其他元素出现多次的情况处理了,这样导致思路受限很多。
解法
方法一(比较法):
11 ms | 40.1 MB | java |
// 思路:先对数组进行排序,然后对 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 ms | 14.9 MB | python3 |
#思路:先对数组排序,显而易见的,单独出现一次的数据必然是出现在数组下标为偶数的位置(下标从0开始),
#那么所有奇数下标的元素之和减去偶数下标的元素之和,就是需要求得的结果。
class Solution:
def singleNumber(self, nums: List[int]) -> int:
nums.sort()
return sum(nums[0::2])-sum(nums[1::2])
方法三(集合法):
56 ms | 14.9 MB | python3 |
#思路:根据set不重复的特点,将所有元素取出求和再乘二,最后减去原始列表,运算结果就是题目的答案。
class Solution:
def singleNumber(self, nums: List[int]) -> int:
return 2*sum(set(nums))-sum(nums)
方法四(异或法):
56 ms | 14.7 MB | python3 |
#思路:根据异或运算的特点,相同的数字经过异或运算后结果为0,除单独出现一次的数字外,
#其他数字都是出现两次的,那么这些数字经过异或运算后结果一定是0。而任何数字与0异或都是本身。
#异或也满足交换率,所以对数组所有元素进行异或运算,运算结果就是题目的答案。
class Solution:
def singleNumber(self, nums: List[int]) -> int:
num = 0
for n in nums:
num = num^n
return num