原题:
数组nums
包含从0
到n
的所有整数,但其中缺了一个。请编写代码找出那个缺失的整数。你有办法在O(n)时间内完成吗?
注意:本题相对书上原题稍作改动
解题思路一:(排序找不同)
数组nums
包含从0
到n
的所有整数,缺失的就在其中,所以我们可以进行排序,使数组的下标与数字对应,所以我们需要找到的是下标和数字不对应的那个,它就是我们要找的缺失的数字。
代码实现:
1. class Solution { 2. public int missingNumber(int[] nums) { 3. 4. Arrays.sort(nums); 5. 6. int i=0; 7. for(i=0;i<nums.length;i++){ 8. 9. if(i!=nums[i]){ 10. break; 11. } 12. } 13. 14. return i; 15. } 16. }
解题思路二:(数字加减法)
用没有缺失的数组中所有数字之和减去缺失了数字的数组所有的数字之和,它们的差就是该缺失的数字。
代码实现:
1. class Solution { 2. public int missingNumber(int[] nums) { 3. int sum=0;//记录没有缺失的和 4. int ret=0;//记录缺失的和 5. for(int i=0;i<nums.length+1;i++){ 6. ret+=i; 7. } 8. 9. for(int i=0;i<nums.length;i++){ 10. sum+=nums[i]; 11. } 12. 13. return ret-sum; 14. 15. } 16. }
解题思路三:(异或法)
a ^ a = 0
; 一个数异或自己,结果为零。a ^ 0 = a
; 一个数异或零,结果为它本身
异或运算满足交换律,结合律 :
a ^ b = b ^ a ; (a ^ b) ^ c = a ^ ( b ^ c ) ;
所以,若 s = a^b^c^d; 如果其中两个数相等,比如 b==c, 则“抵消”,s = a^d。
所以当我们把所有数组下标都异或,并且再异或所有数组中的数字就可以找到缺失的数字,别忘了补上没有缺失的数组的n。
代码实现:
1. class Solution { 2. public int missingNumber(int[] nums) { 3. int ret=0;//用来进行异或运算 4. 5. //对所有下标和数组中的数字进行异或 6. for(int i=0;i<nums.length;i++){ 7. ret^=i; 8. ret^=nums[i]; 9. } 10. //最后别忘了n下标(补上缺失的数字后的数组下标比缺失的多1) 11. ret^=nums.length;//n就是nums.length 12. 13. return ret; 14. 15. } 16. }
原题:
给你一个数组,将数组中的元素向右轮转 k
个位置,其中 k
是非负数。
解题思路: (翻转法)
过程:
代码实现:
1. class Solution { 2. public void rotate(int[] nums, int k) { 3. k%=nums.length;//防止k超出数组范围 4. reverse(nums,0,nums.length-1);//整体翻转 5. reverse(nums,0,k-1);//前部分翻转 6. reverse(nums,k,nums.length-1);//后部分翻转 7. 8. 9. 10. } 11. //自定义翻转方法 12. public void reverse(int[] nums,int left,int right){ 13. 14. while(left<right){ 15. int temp=nums[left]; 16. nums[left]=nums[right]; 17. nums[right]=temp; 18. left++; 19. right--; 20. } 21. 22. } 23. }
空间复杂度为 O(1)