Leetcode-找数专题

个人github:https://github.com/xiongAlen?tab=repositories

  • 1.leetcode 268. 缺失数字
  • 2.leetcode 287. 寻找重复数
  • 3.leetcode 41. 缺失的第一个正数

1.leetcode 268. 缺失数字

题目描述:
给定一个包含 0, 1, 2, ..., n 中 n 个数的序列,找出 0 .. n 中没有出现在序列中的那个数。

思路1:排序。

分析:首先对数组进行排序,然后对顺序数组双端进行检查,即:判断0是否在数组首位,n是否在数组末尾。若两种情况都满足,则确实数字在0-n之间。最后遍历数组即可。

代码(来自官方题解)

class Solution {
    public int missingNumber(int[] nums) {
        //排序
        Arrays.sort(nums);

        // 判断 n 是否出现在末位
        if (nums[nums.length-1] != nums.length) {
            return nums.length;
        }
        // 判断 0 是否出现在首位
        else if (nums[0] != 0) {
            return 0;
        }

        // 此时缺失的数字一定在 (0, n) 中
        for (int i = 1; i < nums.length; i++) {
            int expectedNum = nums[i-1] + 1;
            if (nums[i] != expectedNum) {
                return expectedNum;
            }
        }

        // 未缺失任何数字(保证函数有返回值)
        return -1;
    }
}

思路2:n项和

分析:求n项和减去数组累加和即是所求数。

代码:

class Solution {
    public int missingNumber(int[] nums) {
        int result = 0;
        len=nums.length;
        for (int i = 1; i <= len; i++) {
            result += i - nums[i-1];
        }
        return result;
    }
}

2.leetcode 287. 寻找重复数

题目描述:给定一个包含 n + 1 个整数的数组 nums,其数字都在 1 到 n 之间(包括 1 和 n),可知至少存在一个重复的整数。假设只有一个重复的整数,找出这个重复的数。

思路1:集合

分析:将数组中的数放入集合中(Set),并检查该数是否已存在于集合。若是,则返回该数。

代码:

class Solution {
    public int findDuplicate(int[] nums) {
        Set set = new HashSet();
        for (int num : nums) {
            if (set.contains(num)) {
                return num;
            }
            set.add(num);
        }
        
        return -1;
    }
}

思路2:快慢指针

预备知识:
运用快慢指针寻找循环链表入口节点。

分析:本题可将数组配合下标,抽象称为寻找循环链表入口的问题。例如数组a=[1, 3, 4, 2, 5, 4];元素1的下一个节点为对应数组下标为1的元素。这样抽象成链表list:1->3->2->4->5->4(进入循环);

代码:

class Solution {
    public int findDuplicate(int[] nums) {
        int slow = 0;
        int fast = 0;
        while(slow != fast || fast == 0){
            slow = nums[slow];
            fast = nums[nums[fast]];
        }
        
        //慢指针指向起点
        slow = 0;
        while(fast != slow){
            fast = nums[fast];
            
            
            slow = nums[slow];
        }
        return slow;
    }
}

3.leetcode 41. 缺失的第一个正数

题目描述:给定一个未排序的整数数组,找出其中没有出现的最小的正整数。

思路:桶排序

分析:将数组中的元素打散,并归置于原数组,打散的规则是:将i放于下标为i-1的位置。这样排序后,遍历找出不符合规则的元素,返回其下标+1即所得。

代码:

public class Solution {

    // 关键字:桶排序,什么数字就要放在对应的索引上,其它空着就空着
    // 最好的例子:[3,4,-1,1]
    // 整理好应该是这样:[1,-1,3,4],
    // 这里 1,3,4 都在正确的位置上,
    // -1 不在正确的位置上,索引是 1 ,所以返回 2

    public int firstMissingPositive(int[] nums) {
        int len = nums.length;

        for (int i = 0; i < len; i++) {
            // 注意:先判断给数是否能够构成索引。
            // 再判断该数对应索引的位置上是否为这个数
            // 即 nums[i] ?= nums[nums[i]-1] 这里要特别小心
            while (nums[i] > 0 && nums[i] <= len && nums[nums[i] - 1] != nums[i]) {
                // 第 3 个条件不成立的索引的部分是 i 和 nums[i]-1
                swap(nums, nums[i] - 1, i);
            }
        }

        for (int i = 0; i < len; i++) {
            // [1,-2,3,4]
            // 除了 -2 其它都满足: i+1 = num[i]
            if (nums[i] - 1 != i) {
                return i + 1;
            }
        }

        return len + 1;
    }

    private void swap(int[] nums, int index1, int index2) {
        nums[index1] = nums[index1] ^ nums[index2];
        nums[index2] = nums[index1] ^ nums[index2];
        nums[index1] = nums[index1] ^ nums[index2];
    }
}

优化:上述代码采用位运算交换两元素,替换了加减交换的做法。

参考:https://leetcode-cn.com/problems/first-missing-positive/solution/tong-pai-xu-python-dai-ma-by-liweiwei1419

你可能感兴趣的:(Leetcode-找数专题)