LeetCode习题集 有些题可能直接略过了,整理一下之前刷leetcode

363. 矩形区域不超过 K 的最大数值和

给定一个非空二维矩阵 matrix 和一个整数 k,找到这个矩阵内部不大于 k 的最大矩形和。

示例:

输入: matrix = [[1,0,1],[0,-2,3]], k = 2 输出: 2 解释: 矩形区域 [[0, 1], [-2, 3]] 的数值和是 2,且 2 是不超过 k 的最大数字(k = 2)。 说明:

矩阵内的矩形区域面积必须大于 0。 如果行数远大于列数,你将如何解答呢?

class Solution {

 
public static int maxSumSubmatrix(int[][] matrix, int k) {
    int m = matrix.length;
    int n = matrix[0].length;
    int[][] sums = new int[m][n];

    /**
     * 按照列来求和
     */
    for(int col = 0 ; col < n; col ++){
        for(int row = 0 ; row < m ; row ++){
            if(row == 0)
                sums[row][col] = matrix[row][col];
            else
                sums[row][col] = sums[row - 1][col] + matrix[row][col];
        }
    }

    /**
     * col1为矩阵起始列,col2为矩阵结尾列
     */
    int result = Integer.MIN_VALUE ;
     for(int col1 = 0 ; col1 < n ; col1 ++){
        for(int col2 = col1; col2 < n ; col2++){
            /**
             * set中都是存放的是startCol和endCol相同的矩阵的和
             */
            TreeSet<Integer> set = new TreeSet<>();
            set.add(0);
            for(int row = 0 ; row < m ; row++){
                int sum = 0;  //子矩阵的和
                for(int i = col1; i <= col2 ; i ++){
                    sum += sums[row][i];
                }

                //求出set中大于等于(sum - k)最小值
                if(set.ceiling(sum - k) != null){
                    int max = sum - set.ceiling(sum - k);
                    result = result > max ? result : max;
                }
                set.add(sum);
            }
        }
    }
    return result;
}
}

365. 水壶问题

有两个容量分别为 x升 和 y升 的水壶以及无限多的水。请判断能否通过使用这两个水壶,从而可以得到恰好 z升 的水?

如果可以,最后请用以上水壶中的一或两个来盛放取得的 z升 水。

你允许:

装满任意一个水壶 清空任意一个水壶 从一个水壶向另外一个水壶倒水,直到装满或者倒空 示例 1: (From the famous "Die Hard" example)

输入: x = 3, y = 5, z = 4 输出: True 示例 2:

输入: x = 2, y = 6, z = 5 输出: False

	如果z=a*x+b*y(a,b均整),x与y最大公约数为g,那么z一定是g的整数倍,即z%g=0!!

class Solution {
    public boolean canMeasureWater(int x, int y, int z) {
   return z == 0 || (x + y >= z && z % gcd(x, y) == 0);
    }
    int gcd(int x, int y) {
        return y == 0 ? x : gcd(y, x % y);
    }

   
}

367. 有效的完全平方数

给定一个正整数 num,编写一个函数,如果 num 是一个完全平方数,则返回 True,否则返回 False。

说明:不要使用任何内置的库函数,如 sqrt。

示例 1:

输入:16 输出:True 示例 2:

输入:14 输出:False

	牛顿迭代法
        如果大于就找x+num对x的因子的一半
        逐渐趋近于完全平方数,如果x*x小于了num还没找到,证明不是完全平方
class Solution {
    public boolean isPerfectSquare(int num) { 
          if (num < 2) return true;
        long x = num;
        while (x * x > num) {
            x = (x + num / x) / 2;
            if (x * x == num) {
                return true;
            }
        }
        return false;
    }
}

368. 最大整除子集

给出一个由无重复的正整数组成的集合,找出其中最大的整除子集,子集中任意一对 (Si,Sj) 都要满足:Si % Sj = 0 或 Sj % Si = 0。

如果有多个目标子集,返回其中任何一个均可。

示例 1:

输入: [1,2,3] 输出: [1,2] (当然, [1,3] 也正确) 示例 2:

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

class Solution {
  public int max(int a,int b){
        return a>b?a:b;
    }
    public List<Integer> largestDivisibleSubset(int[] nums) {
        int n=nums.length;
        if(n==0)
            return new ArrayList();
        int[] last=new int[n];
        for(int i=0;i<n;i++)
            last[i]=-1;
        int[] dp=new int[n];
        int ans=0;
        Arrays.sort(nums);
        for(int i=0;i<n;i++)
            for(int j=0;j<i;j++){
                if(nums[i]%nums[j]==0&&dp[j]+1>dp[i]){
                    dp[i]=dp[j]+1;
                    if(dp[ans]<dp[i])
                        ans=i;
                    last[i]=j;
                }
            }
        List<Integer> ansList=new ArrayList();
        while(ans!=-1){
            ansList.add(nums[ans]);
            ans=last[ans];
        }
        return ansList;
    }
}

371. 两整数之和

不使用运算符 + 和 - ​​​​​​​,计算两整数 ​​​​​​​a 、b ​​​​​​​之和。

示例 1:

输入: a = 1, b = 2 输出: 3 示例 2:

输入: a = -2, b = 3 输出: 1 PS:

   sum = a ^ b;  //异或这里可看做是相加但是不显现进位,比如5 ^ 3
                 /*0 1 0 1
                   0 0 1 1
                 ------------
                   0 1 1 0      
              上面的如果看成传统的加法,不就是1+1=2,进1得0,但是这里没有显示进位出来,仅是相加,0+1或者是1+0都不用进位*/
    
    carry = (a & b) << 1;
    
                //相与为了让进位显现出来,比如5 & 3
                /* 0 1 0 1
                   0 0 1 1
                 ------------
                   0 0 0 1
              上面的最低位1和1相与得1,而在二进制加法中,这里1+1也应该是要进位的,所以刚好吻合,但是这个进位1应该要再往前一位,所以左移一位*/
    
class Solution {
    public int getSum(int a, int b) {
 return b == 0 ? a : getSum(a^b,(a&b)<<1); 
    }
}

372. 超级次方

你的任务是计算 ab 对 1337 取模,a 是一个正整数,b 是一个非常大的正整数且会以数组形式给出。

示例 1:

输入: a = 2, b = [3] 输出: 8 示例 2:

输入: a = 2, b = [1,0] 输出: 1024

PS: 我现在饱受数学的折磨,欧拉筛,欧拉函数,

(((φ(◎ロ◎;)φ)))

class Solution {
     //372超级次方
    public int superPow(int a ,int[] b){
        int c = 1337;
        int exp = 0;
        int phi = euler(c);
        //同余
        for(int i=0;i<b.length;i++) exp = (exp*10+b[i])%1140;

        return qucikPow(a%1337,exp,1337);
    }
    //快速幂计算
    public int qucikPow(int a,int b,int c){
        int ans =1;
        while(b>0){
            if((b&1)==1){
                ans = (ans*a)%c;
            }
            b >>=1;
            a = (a*a)%c;
        }
        return ans;
    }
    //欧拉函数
    public int euler(int n){
        int ret= n;
        for(int i=2;i*i<n;i++){
            if(n%i==0){//n的质因数
                ret = ret/n*(n-1); //欧拉函数公式
                while(n%i==0){//去掉质因数i
                    n/=i;
                }
            }
        }
        if(n>1){//n本来就是质数 f(n) = n-1;
            ret = ret/n*(n-1);
        }
        return ret;
    }

}

373. 查找和最小的K对数字

给定两个以升序排列的整形数组 nums1 和 nums2, 以及一个整数 k。

定义一对值 (u,v),其中第一个元素来自 nums1,第二个元素来自 nums2。

找到和最小的 k 对数字 (u1,v1), (u2,v2) ... (uk,vk)。

示例 1:

输入: nums1 = [1,7,11], nums2 = [2,4,6], k = 3
输出: [1,2],[1,4],[1,6]
解释: 返回序列中的前 3 对数:

 [1,2],[1,4],[1,6],[7,2],[7,4],[11,2],[7,6],[11,4],[11,6]

示例 2:

输入: nums1 = [1,1,2], nums2 = [1,2,3], k = 2
输出: [1,1],[1,1]
解释: 返回序列中的前 2 对数:
     [1,1],[1,1],[1,2],[2,1],[1,2],[2,2],[1,3],[1,3],[2,3]

示例 3:

输入: nums1 = [1,2], nums2 = [3], k = 3 
输出: [1,3],[2,3]
解释: 也可能序列中所有的数对都被返回:[1,3],[2,3]
indexArray的标记每一次都会加加,以至于是不能找到重复的,
class Solution {
     public List<List<Integer>> kSmallestPairs(int[] nums1, int[] nums2, int k) {
        List<List<Integer>> result = new ArrayList<>();
        if (k <= 0) {
            return result;
        }
        // 搜索的起始位置,主要起到优化作用减少循环次数
        int[] indexArray = new int[nums1.length];
        int startIndex = 0;
        while (result.size() < k) {

            // 记录当前最小对的和
            int min = Integer.MAX_VALUE;

            // 记录当前最小对的位置
            int currentIndex = -1;
            for (int i = startIndex; i < nums1.length; i++) {
                // 这里说明nums1[i]已经与nums2中所有元素结对入队列,之后的搜索从nums1[i + 1]开始
                if (indexArray[i] == nums2.length) {
                    startIndex = i + 1;
                    continue;
                }

                // 比较,选择最小的
                if (nums1[i] + nums2[indexArray[i]] < min) {
                    min = nums1[i] + nums2[indexArray[i]];
                    currentIndex = i;
                }

                // nums1和nums2都升序,所以nums1[i] + nums2[a] <= nums1[nums1.length - 1] + nums2[a]
                if (indexArray[i] == indexArray[indexArray.length - 1]) {
                    break;
                }
            }

            // 防止k > nums1.length * nums2.length,出现则直接跳出
            if (currentIndex == -1) {
                break;
            }

            // 最小的对入队
            List<Integer> data = new ArrayList<>();
            result.add(data);
            data.add(nums1[currentIndex]);
            data.add(nums2[indexArray[currentIndex]]);
            indexArray[currentIndex] = indexArray[currentIndex] + 1;
        }

        return result;
    }
}

374. 猜数字大小

我们正在玩一个猜数字游戏。 游戏规则如下: 我从 1 到 n 选择一个数字。 你需要猜我选择了哪个数字。 每次你猜错了,我会告诉你这个数字是大了还是小了。 你调用一个预先定义好的接口 guess(int num),它会返回 3 个可能的结果(-1,1 或 0):

-1 : 我的数字比较小 1 : 我的数字比较大 0 : 恭喜!你猜对了! 示例 :

输入: n = 10, pick = 6 输出: 6 PS: 强烈建议力扣换一个描述的,这个题描述的,简直是………… 我的数字指的是要猜的数字,并不是你输入的数字

/* The guess API is defined in the parent class GuessGame.
   @param num, your guess
   @return -1 if my number is lower, 1 if my number is higher, otherwise return 0
      int guess(int num); */

public class Solution extends GuessGame {
    public int guessNumber(int n) {
          long l = 1, r=n;
        int res;
        while((res = guess((int)((l+r)/2))) != 0){
            if(-1 == res){
                r = (l+r)/2-1;
            }else if(1 == res){
                l = (l+r)/2+1;
            }          
        }
        return (int)((l+r)/2);
    }
}

375. 猜数字大小 II

我们正在玩一个猜数游戏,游戏规则如下:

我从 1 到 n 之间选择一个数字,你来猜我选了哪个数字。

每次你猜错了,我都会告诉你,我选的数字比你的大了或者小了。

然而,当你猜了数字 x 并且猜错了的时候,你需要支付金额为 x 的现金。直到你猜到我选的数字,你才算赢得了这个游戏。

示例:

n = 10, 我选择了8.

第一轮: 你猜我选择的数字是5,我会告诉你,我的数字更大一些,然后你需要支付5块。 第二轮: 你猜是7,我告诉你,我的数字更大一些,你支付7块。 第三轮: 你猜是9,我告诉你,我的数字更小一些,你支付9块。

游戏结束。8 就是我选的数字。

你最终要支付 5 + 7 + 9 = 21 块钱。 给定 n ≥ 1,计算你至少需要拥有多少现金才能确保你能赢得这个游戏。

PS:可能是我还没处理太好,效率还是有些低还望大佬提供一下思路

        dp[i][j]表示从[i,j]中猜出正确数字所需要的最少花费金额.(dp[i][i] = 0)
        假设在范围[i,j]中选择x, 则选择x的最少花费金额为: max(dp[i][x-1], dp[x+1][j]) + x
        用max的原因是我们要计算最坏反馈情况下的最少花费金额(选了x之后, 正确数字落在花费更高的那侧)
        
        初始化为(n+2)*(n+2)数组的原因: 处理边界情况更加容易, 例如对于求解dp[1][n]时x如果等于1, 需要考虑dp[0][1](0不可能出现, dp[0][n]为0)
        而当x等于n时, 需要考虑dp[n+1][n+1](n+1也不可能出现, dp[n+1][n+1]为0)
        
        如何写出相应的代码更新dp矩阵, 递推式dp[i][j] = max(max(dp[i][x-1], dp[x+1][j]) + x), x~[i:j], 可以画出矩阵图协助理解, 可以发现
        dp[i][x-1]始终在dp[i][j]的左部, dp[x+1][j]始终在dp[i][j]的下部, 所以更新dp矩阵时i的次序应当遵循bottom到top的规则, j则相反, 由于
        i肯定小于等于j, 所以我们只需要遍历更新矩阵的一半即可(下半矩阵)
class Solution {
     public int getMoneyAmount(int n) {
       
        int[][] dp = new int[n+2][n+2];
        for(int i = n; i >= 1; --i) {
            for(int j = i; j <= n; ++j) {
                if(i == j)
                    dp[i][j] = 0;
                else {
                    dp[i][j] = Integer.MAX_VALUE;
                    for(int x = i; x <= j; ++x) 
                        dp[i][j] = Math.min(dp[i][j], Math.max(dp[i][x-1], dp[x+1][j]) + x);
                }
            }
        }
        return dp[1][n];
    }
}

376. 摆动序列

如果连续数字之间的差严格地在正数和负数之间交替,则数字序列称为摆动序列。第一个差(如果存在的话)可能是正数或负数。少于两个元素的序列也是摆动序列。

例如, [1,7,4,9,2,5] 是一个摆动序列,因为差值 (6,-3,5,-7,3) 是正负交替出现的。相反, [1,4,7,2,5] 和 [1,7,4,5,5] 不是摆动序列,第一个序列是因为它的前两个差值都是正数,第二个序列是因为它的最后一个差值为零。

给定一个整数序列,返回作为摆动序列的最长子序列的长度。 通过从原始序列中删除一些(也可以不删除)元素来获得子序列,剩下的元素保持其原始顺序。

示例 1:

输入: [1,7,4,9,2,5] 输出: 6 解释: 整个序列均为摆动序列。 示例 2:

输入: [1,17,5,10,13,15,10,5,16,8] 输出: 7 解释: 这个序列包含几个长度为 7 摆动序列,其中一个可为[1,17,10,13,10,16,8]。 示例 3:

输入: [1,2,3,4,5,6,7,8,9] 输出: 2 进阶: 你能否用 O(n) 时间复杂度完成此题?

class Solution {
     public int wiggleMaxLength(int[] nums) {
        int n = nums.length;
        if (n < 2) {
            return n;
        }
        int up = 1;
        int down = 1;
        for (int i = 1; i < n; i++) {
            if (nums[i] > nums[i - 1]) {
                up = down + 1;
            }
            if (nums[i] < nums[i - 1]) {
                down = up + 1;
            }
        }
        return Math.max(up, down);
    }
}

377. 组合总和 Ⅳ

给定一个由正整数组成且不存在重复数字的数组,找出和为给定目标正整数的组合的个数。

示例:

nums = [1, 2, 3] target = 4

所有可能的组合为: (1, 1, 1, 1) (1, 1, 2) (1, 2, 1) (1, 3) (2, 1, 1) (2, 2) (3, 1)

请注意,顺序不同的序列被视作不同的组合。

因此输出为 7。 进阶: 如果给定的数组中含有负数会怎么样? 问题会产生什么变化? 我们需要在题目中添加什么限制来允许负数的出现?

致谢: 特别感谢 @pbrother 添加此问题并创建所有测试用例。

class Solution {
      public int combinationSum4(int[] nums, int target) {
        int[] memo = new int[target + 1];
        memo[0] = 1;
        for (int i = 0; i < target; i++) {
            for (int num : nums) {
                if (i + num <= target) {
                    memo[i + num] += memo[i];
                }
            }
        }
        return memo[target];
    }
}

378. 有序矩阵中第K小的元素

给定一个 n x n 矩阵,其中每行和每列元素均按升序排序,找到矩阵中第k小的元素。 请注意,它是排序后的第k小元素,而不是第k个元素。

示例:

matrix = [
   [ 1,  5,  9],
   [10, 11, 13],
   [12, 13, 15]
],
k = 8,

返回 13。 说明: 你可以假设 k 的值永远是有效的, 1 ≤ k ≤ n2 。

class Solution {
     public int kthSmallest(int[][] matrix, int k) {
        int n = matrix.length;
        int left = matrix[0][0], right = matrix[n-1][n-1];
        while(left < right){
            int mid = (left + right)/2;
            int cnt = numMid(matrix, n, mid);
            //System.out.println(cnt);
            if(cnt >= k){
                right = mid ;
            } else {
                left = mid + 1;
            }
        }
        return right;
    }

    int numMid(int[][] matrix, int n, int mid){
        int i = 0, j = n - 1;
        int cnt = 0;
        while(i<n&& j >=0){
            if(matrix[i][j] > mid){
                j--;
            } else {
                cnt += (j+1);
                i++;
            }
        }
        return cnt;
    }
}

380. 常数时间插入、删除和获取随机元素

设计一个支持在平均 时间复杂度 O(1) 下,执行以下操作的数据结构。

insert(val):当元素 val 不存在时,向集合中插入该项。
remove(val):元素 val 存在时,从集合中移除该项。
getRandom:随机返回现有集合中的一项。每个元素应该有相同的概率被返回。

示例 :

// 初始化一个空的集合。
RandomizedSet randomSet = new RandomizedSet();

// 向集合中插入 1 。返回 true 表示 1 被成功地插入。
randomSet.insert(1);

// 返回 false ,表示集合中不存在 2 。
randomSet.remove(2);

// 向集合中插入 2 。返回 true 。集合现在包含 [1,2] 。
randomSet.insert(2);

// getRandom 应随机返回 1 或 2 。
randomSet.getRandom();

// 从集合中移除 1 ,返回 true 。集合现在包含 [2] 。
randomSet.remove(1);

// 2 已在集合中,所以返回 false 。
randomSet.insert(2);

// 由于 2 是集合中唯一的数字,getRandom 总是返回 2 。
randomSet.getRandom();
class RandomizedSet {

    /**
        核心思路是用map记录数组每个值对应的下标,在插入和删除时不需要遍历数组寻找对应的数字下标,从而实现插入O(1)
        由于删除数组中的数据不是O(1),这里将数组中待删除的数字用数组最后一个数字代替,然后size - 1,下次insert的时候将size位置的数字覆盖,从而实现O(1)删除
    **/
    private Map<Integer,Integer> map;
    private List<Integer> list;
    private int size;
    /** Initialize your data structure here. */
    public RandomizedSet() {
        map = new HashMap<>();
        list = new ArrayList<>();
        size = 0;
    }
    
    /** Inserts a value to the set. Returns true if the set did not already contain the specified element. */
    public boolean insert(int val) {
        if(map.containsKey(val)){
            return false;
        }else{
            //这里需要用add(index,value)的重载方法,覆盖当前的size位置,这个方法在size == index也可以使用
            //不能使用set(index,value)方法,这个方法会在size == index时直接抛异常
            list.add(size,val);
            map.put(val,size);
            size++;
        }
        return true;
    }
    
    /** Removes a value from the set. Returns true if the set contained the specified element. */
    public boolean remove(int val) {
        if(!map.containsKey(val)){
            return false;
        }else{
            int oldIndex = map.get(val);
            int lastVal = list.get(size - 1);
            list.set(oldIndex,lastVal);
            map.put(lastVal,oldIndex);
            map.remove(val);
            size--;
        }
        return true;
    }
    
    /** Get a random element from the set. */
    public int getRandom() {
        Random ran = new Random();
        return list.get(ran.nextInt(size));
    }
}

/**
 * Your RandomizedSet object will be instantiated and called as such:
 * RandomizedSet obj = new RandomizedSet();
 * boolean param_1 = obj.insert(val);
 * boolean param_2 = obj.remove(val);
 * int param_3 = obj.getRandom();
 */