Leetcode 刷题册

 

LingLing practices 40 hours per day. How about you?

动态规划

很多动态规划问题大致可以按照下面5个步骤取解释: 1、状态定义 2、状态转移方程 3、初始化 4、输出 5、思考状态压缩

动态规划原理

都是把大问题拆分成小问题。寻找大问题和小问题的递推关系。解决一个个小问题,最终达到解决原问题的效果。动态规划有记忆性,填表把所有已经解决的子问题答案记录下来,在新问题里需要用到的子问题可以直接提取,避免了重复计算,从而节约了时间,所以在问题满足最优性原理之后,用动态规划解决问题的核心就在于填表,表填写完毕,最优解也就找到。

198 打家劫舍

如果房间数量 大于 2间。偷前k间房屋的最大金额。

  • 如果偷第k间屋子,就不能偷第k-1间屋子,总金额为前k-2间房屋的最高总金额和第k间房屋的金额之和。
  • 不偷第k间屋子,总金额为偷前k-1间屋子的最高金额。

用dp[i]表示偷前i间屋子可以偷到的最高金额。

class Solution {
    public int rob(int[] nums) {
        if(nums == null || nums.length == 0)
            return 0;
        int length = nums.length;
        if(length == 1)
            return nums[0];
        int[] dp = new int[length];
        dp[0] = nums[0];
        dp[1] = Math.max(nums[0], nums[1]);
        for(int i=2;i<length;i++){
            dp[i] = Math.max(dp[i-1], dp[i-2]+nums[i]);
        }
        return dp[length-1];
    }
}

213 打家劫舍2

题目描述:房屋都围成一圈,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。

解题思路:在打家劫舍1的基础上,说给出的房子是环形排列首尾相接的。区别就是第一栋房子和最后一栋房子不能同时偷。只需要计算两种情况:

  1. 不偷第一个,问题就是在后边n-1个房子可以偷的最大价值。
  2. 不偷最后一个,问题就是在前边n-1个房子可以偷的最大价值。

这样转换就没有首尾连接的约束了,可以直接采用上一题的解法。取两种情况的最大值

class Solution {
public:
    //首尾之能选一个偷
    int rob(vector<int>& nums) {
        //思想很简单,不偷第一家,就在后面n-1家里偷,不偷最后一家,就在前n-1家里偷
        int len=nums.size();
        if(len==0)
            return 0;
        if(len==1)
            return nums[0];

        int ans1 = rob1(nums, 0, len-2);
        int ans2 = rob1(nums, 1, len-1);

        return max(ans1,ans2);
    }

    int rob1(vector<int> &nums, int l, int r){

        int ans = 0;
        int pre=0,prepre=0;

        for(int i=l;i<=r;i++){
            ans = max(prepre+nums[i], pre);
            prepre = pre;
            pre = ans;
        }
        return ans;
    }
};

213 打家劫舍3

题目描述:房屋成一个二叉树结构,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。

解题思路:结点r,分为偷和不偷两种情况。

偷r结点:左右儿子不能偷。最大价值等于不偷左儿子的最大价值+不偷右儿子的最大价值+r的价值

不偷r节点: 最大价值等于偷或不偷左儿子的最大价值+偷或不偷右儿子的最大价值。

class Solution {
public:
    typedef pair<int, int> pii;
    int rob(TreeNode* root) {
        // 可行窃地区是二叉树
        if(root==nullptr)
            return 0;
        pii ans = helper(root);
        return max(ans.first, ans.second);

    }
    //偷r结点:左右儿子不能偷。最大价值等于不偷左儿子的最大价值+不偷右儿子的最大价值+r的价值

    //不偷r节点: 最大价值等于**偷或不偷**左儿子的最大价值+**偷或不偷**右儿子的最大价值。
    pii helper(TreeNode* root){
        if(root==nullptr)
            return {0,0};
        //helper 返回 偷 这个结点能取到的最大价值,还有 不偷 这个结点时能取到的最大价值
        pii l = helper(root->left);
        pii r = helper(root->right);

        //偷
        int r0 = l.second+r.second+root->val;
        //不偷当前
        int r1 = max(l.first,l.second)+max(r.first,r.second);
        return make_pair(r0,r1);
    }
};

740. 删除与获得点数

题目链接

给定一个整数数组 nums ,你可以对它进行一些操作。 每次操作中,选择任意一个 nums[i] ,删除它并获得 nums[i] 的点数。之后,你必须删除每个等于 nums[i] - 1 或 nums[i] + 1 的元素。 开始你拥有 0 个点数。返回你能通过这些操作获得的最大点数。

解题思路:

转换成打家劫舍:我们在原来的 nums 的基础上构造一个临时的数组 all,这个数组,以元素的值来做下标,下标对应的元素是原来的元素的个数。 举个例子: nums = [2, 2, 3, 3, 3, 4] 构造后: all=[0, 0, 2, 3, 1]; 就是代表着 22 的个数有两个,33 的个数有 33 个,44 的个数有 11 个。 其实这样就可以变成打家劫舍的问题了

class Solution {
    public int deleteAndEarn(int[] nums) {
        int n = nums.length;
        if(n==0) return 0;
        int max = 0;
        for(int num: nums){
            max = Math.max(max, num);
        }
        int[] all = new int[max+1];
        for(int i=0; i<n;i++){
            all[nums[i]]++;
        }
        
        int[] dp = new int[max+1];
        if(max==1)
            return all[1];
        if(max==2)
            return Math.max(all[1], 2*all[2]);
        
        dp[1] = all[1];
        dp[2] = Math.max(all[1], 2*all[2]);
        for(int i=3;i<=max;i++){
            dp[i] = Math.max(dp[i-2]+all[i]*i,dp[i-1]);
        }
        return dp[max];
    }
}

72 编辑距离

题目描述:删除,修改,插入,把word1变成word2的最小编辑距离

思路:动态规划,D[i][j]表示word1的前i个字母和word2的前j个之间的编辑距离。

  • D[i][j-1] 为 A 的前 i 个字符和 B 的前 j - 1 个字符编辑距离的子问题。即对于 B 的第 j 个字符,我们在 A 的末尾添加了一个相同的字符,那么 D[i][j]最小可以为 D[i][j-1] + 1

  • D[i-1][j] 为 A 的前 i - 1 个字符和 B 的前 j 个字符编辑距离的子问题。即对于 A 的第 i 个字符,我们在 B 的末尾添加了一个相同的字符,那么 D[i][j] 最小可以为 D[i-1][j] + 1

  • D[i-1][j-1] 为 A 前 i - 1 个字符和 B 的前 j - 1 个字符编辑距离的子问题。即对于 B 的第 j 个字符,我们修改 A 的第 i 个字符使它们相同,那么 D[i][j] 最小可以为 D[i-1][j-1] + 1。特别地,如果 A 的第 i 个字符和 B 的第 j 个字符原本就相同,那么我们实际上不需要进行修改操作。在这种情况下,D[i][j] 最小可以为 D[i-1][j-1]

如果word1[i]==word2[j]: \(\begin{aligned} D[i][j] &=\min (D[i][j-1]+1, D[i-1][j]+1, D[i-1][j-1]) \\ &=1+\min (D[i][j-1], D[i-1][j], D[i-1][j-1]-1) \end{aligned}\) 如果word1[i]!=word2[j]: \(D[i][j]=1+\min (D[i][j-1], D[i-1][j], D[i-1][j-1])\)

    int minDistance(string word1, string word2) {
        int n = word1.length();
        int m = word2.length();

        // 有一个字符串为空串
        if (n * m == 0) return n + m;

        // DP 数组
        int D[n + 1][m + 1];

        // 边界状态初始化
        for (int i = 0; i < n + 1; i++) {
            D[i][0] = i;
        }
        for (int j = 0; j < m + 1; j++) {
            D[0][j] = j;
        }

        // 计算所有 DP 值
        for (int i = 1; i < n + 1; i++) {
            for (int j = 1; j < m + 1; j++) {
                int left = D[i - 1][j] + 1;
                int down = D[i][j - 1] + 1;
                int left_down = D[i - 1][j - 1];
                if (word1[i - 1] != word2[j - 1]) left_down += 1;
                D[i][j] = min(left, min(down, left_down));

            }
        }
        return D[n][m];
    }

32. 最长有效括号 7月4日

题目链接 给定一个只包含 ‘(‘ 和 ‘)’ 的字符串,找出最长的包含有效括号的子串的长度。

栈的思想,栈底是「最后一个没有被匹配的右括号的下标」

class Solution {
    public int longestValidParentheses(String s) {
        //最长有效括号
        //栈的思想的话,栈底是「最后一个没有被匹配的右括号的下标」
        Stack<Integer> stc = new Stack<>();
        stc.push(-1);
        int len = s.length();
        int ans = 0;
        for(int i=0;i<len;i++){
            if('('==s.charAt(i)){
                stc.push(i);
            }else{
                stc.pop();
                if(stc.empty()){
                    stc.push(i);
                }else{
                    ans = Math.max(ans, i-stc.peek());
                }
            }
        }
        return ans;
    }
}

416 分割等和子集

题目链接

题目描述:给定一个只包含正整数的非空数组。是否可以将这个数组分割成两个子集,使得两个子集的元素和相等。 注意: 每个数组中的元素不会超过 100 数组的大小不会超过 200

解题思路:

动态规划,转换01背包问题 boolean[][] dp 数组初始化的时候默认就是false,dp[i][j] 表示 编号范围在[0,i] 的数字,能够从中选出和为j的数组出来。

2种边界情况:

  • 对于所有的idp[i][0] = true
  • i==0时,dp[0][nums[0]] = true;
class Solution {
    public boolean canPartition(int[] nums) {
        int n = nums.length;
        if (n < 2) {
            return false;
        }
        int sum = 0, maxNum = 0;
        for (int num : nums) {
            sum += num;
            maxNum = Math.max(maxNum, num);
        }
        if (sum % 2 != 0) {
            return false;
        }
        int target = sum / 2;
        if (maxNum > target) {
            return false;
        }
        boolean[][] dp = new boolean[n][target + 1]; //初始状态 dp全部都是`false`
        for (int i = 0; i < n; i++) {
            dp[i][0] = true;
        }
        dp[0][nums[0]] = true;
        for(int i=1; i< n; i++){
            int num = nums[i];
            for(int j=1; j <= target; j++){
                if(j>=num)
                    dp[i][j] = dp[i-1][j-num] | dp[i-1][j];
                else
                    dp[i][j] = dp[i-1][j];
            }
        }
        return dp[n-1][target];
    }
}

125. 乘积最大子数组

题目链接

因为乘法有正负数的情况,子问题可能不是从i-1这个状态变过来的。 也没有用复杂的动态规划,就是在维护一个数组记录以nums[i]结尾的子数组乘积的最小值。

class Solution {
    public int maxProduct(int[] nums) {
        int n = nums.length;
        int[] Fmax = new int[n];
        int[] Fmin = new int[n];
        System.arraycopy(nums, 0, Fmax, 0, n);
        System.arraycopy(nums, 0, Fmin, 0, n);

        for(int i=1;i<n;i++){
            int num = nums[i];
            Fmax[i] = Math.max(Fmax[i-1]*num, Math.max(Fmin[i-1]*num, num));
            Fmin[i] = Math.min(Fmax[i-1]*num, Math.min(Fmin[i-1]*num, num));
        }
        int ans = Fmax[0];
        for(int i=1;i<n;i++){
            ans = Math.max(ans, Fmax[i]);
        }
        return ans;
    }
}

1143. 最长公共子序列

题目链接

class Solution {
    public int longestCommonSubsequence(String text1, String text2) {
        char[] chars1 = text1.toCharArray();
        char[] chars2 = text2.toCharArray();

        int m = text1.length();
        int n = text2.length();
        int[][] dp = new int[m+1][n+1];
        
        for(int i=1; i<=m; i++){
            for(int j=1; j<=n; j++){
                if( chars1[i-1]==chars2[j-1] ){
                    dp[i][j] = dp[i-1][j-1]+1;
                }else{
                    dp[i][j] = Math.max(dp[i-1][j],dp[i][j-1]);
                }
            }
        }
        return dp[m][n];
    }
}

97. 交错字符串

题目链接

class Solution {
    public boolean isInterleave(String s1, String s2, String s3) {
        
        int m = s1.length();
        int n = s2.length();
        int t = s3.length();
        if (n + m != t) {
            return false;
        }
        boolean[][] dp = new boolean[m+1][n+1];

        dp[0][0] = true;

        for(int i=0; i<=m; i++)
            for(int j=0; j<=n; j++)
            {
                int p = i + j - 1;
                if(i > 0){
                    dp[i][j] = dp[i][j] || (dp[i-1][j] && s1.charAt(i-1)==s3.charAt(p));
                }
                if(j > 0)
                {
                    dp[i][j] = dp[i][j] || (dp[i][j-1] && s2.charAt(j-1)==s3.charAt(p));
                }
            }
        return dp[m][n];
    }
}

二叉树非递归遍历

前序遍历非递归

class Solution{
    public void flatten(TreeNode root){
        List<TreeNode> list = new ArrayList<TreeNode>();
        Deque<TreeNode> stack = new LinkedList<TreeNode>();
        TreeNode node = root;
        while(node!=null || !stack.isEmpty()){
            while(node!=null){
                list.add(node);
                stack.push(node);
                node=node.left;
            }
            node=stack.pop();
            node=node.right;
        }
        int size = list.size();
        for(int i=1; i<size;i++){
            TreeNode prev = list.get(i-1);
            prev.left=null;
            prev.right=list.get(i);
        }
    }
}

中序遍历非递归

用辅助栈

class Solution{
    public List<Integer> inorderTraversal(TreeNode root){
        List<Integer> list = new ArrayList<>();
        Stack<Integer> stack = new Stack<>();
        if(root!=null) return list;
        TreeNode node = root;
        while(node!=null || !stack.empty()){
            while(node!=null){
                stack.push(node);
                node=node.left;
            }
            if(!stack.empty()){
                node=stack.pop();
                list.add(node); //访问节点
                node=node.right;
            }
        }
        return list;
    }
}

后序遍历非递归

Leetcode:二叉树的后序遍历

  • 思路一: 依次将根结点,右节点,左节点押入栈。根->右->左,和先序遍历的根->右->左,很像,就把先序的左右交换,访问改成压栈,压另一个栈。
class Solution{
    public List<Integer> postorderTraversal(TreeNode root){
        Stack<Integer> stack = new Stack<>();
        List<Integer> list = new ArrayList<>();
        if(root==null) return list;
        TreeNode node = root;
        while(node!=null || !stack.empty()){
            while(node!=null){
                list.add(node);
                stack.push(node);
                node=node.right;
            }
            if(!stack.empty()){
                node=stack.pop();
                node=node.left;
            }
        }
        Collections.reverse(list);
        return list;
    }
}
class Solution {
    public List<Integer> postorderTraversal(TreeNode root) {
        Deque<TreeNode> stack = new LinkedList<>();
        LinkedList<Integer> ans = new LinkedList<>();
        if (null == root) return ans;
        stack.addFirst(root);
        while(!stack.isEmpty()) {
            TreeNode node = stack.removeFirst();
            ans.addFirst(node.val);
            if (null != node.left) {
                stack.addFirst(node.left);
            }
            if (null != node.right) {
                stack.addFirst(node.right);
            }
        }
        return ans;
    }
}
  • 思路二:

要访问一个节点的条件上一个访问的节点是右儿子。我们可以增加一个变量Prev来判断当前节点Curr的上一个节点与它的关系来执行相应的操作。

  • 若Prev为空(Curr节点是根节点)或者Prev是Curr的父节点,将Curr节点的左孩子和右孩子分别压入栈;
  • 若Prev是Curr的左儿子,则将Curr的右儿子压入栈;
  • 否则Prev是Curr的右儿子,访问Curr;
class Solution{
    public List<Integer> postorderTraversal(TreeNode root){
        Stack<TreeNode> stack = new Stack<>();
        List<Integer> list = new ArrayList<>();
        if(root==null) return list;
        TreeNode prev = null, cur = null;
        stack.push(root);
        while(!stack.empty()){
            cur = stack.peek();//不要弹出
            if(prev==null || prev.left==cur || prev.right==cur){
                if(cur.left!=null)
                    stack.push(cur.left);
                else if(cur.right!=null)
                    stack.push(cur.right);
            }else if(prev == cur.left){
                if(cur.right!=null)
                    stack.push(cur.right);
            }else{
                list.add(cur.val);
                stack.pop();
            }
            prev = cur;
        }
        return list;
    }
}

贪心

55. 跳跃游戏

题目链接

贪心策略,总是跳到“最大下一跳距离最大”的位置

class Solution {
    public boolean canJump(int[] nums) {
        int n = nums.length;
        if(n<=1){
            return true;
        }

        int now = 0;
        int next = 0;
        while(now < n-1){
            int maxDistance = now + nums[now];
            if(maxDistance < now+1) return false;
            next = now+1;
            int max = now+1 + nums[now+1];
            for(int i=now+2; i<=maxDistance && i<n;i++){
                if(i+nums[i] > max){
                    max = i + nums[i];
                    next = i;
                }
            }
            now = next;
        }

        return true;

    }
}

二分查找

left,right。怎么确定每一步left和right的移动方式。 mid = (left+right)/2mid = (left+right+1)/2两种不一样,第一种需要right=mid;else left=mid+1。第二种需要left = mid; else right = mid-1。原因在于计算确定mid的方法有区别。

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

题目链接

二分法是解法之一,每次得到二分数组mid,然后计算矩阵中 <mid的个数。

class Solution {
    public int kthSmallest(int[][] matrix, int k) {
        //二分法查找
        int n = matrix.length;
        if(n<=0){
            return -1;
        }
        
        //假设K永远有效
        int left = matrix[0][0];
        int right = matrix[n-1][n-1];
        int mid;
        while(left<right){
            mid = (left+right)/2;
            if(check(matrix, k, n,mid)){
                right = mid;
            }else{
                left = mid+1;
            }
        }
        return left;
    }

    private boolean check(int[][] matrix, int k, int n, int mid){
        int num=0;
        //左下角开始搜索
        int i=n-1;
        int j=0;
        
        while( i>=0 && j<n ){
            if(matrix[i][j]<=mid){
                j++;
                num+=(i+1);
            }else{
                i--;
            }
        }

        return num>=k;
    }
}

778. 水位上升的泳池中游泳

题目链接

在一个 N x N 的坐标方格 grid 中,每一个方格的值 grid[i][j] 表示在位置 (i,j) 的平台高度。

现在开始下雨了。当时间为 t 时,此时雨水导致水池中任意位置的水位为 t 。你可以从一个平台游向四周相邻的任意一个平台,但是前提是此时水位必须同时淹没这两个平台。假定你可以瞬间移动无限距离,也就是默认在方格内部游动是不耗时的。当然,在你游泳的时候你必须待在坐标方格里面。

你从坐标方格的左上平台 (0,0) 出发。最少耗时多久你才能到达坐标方格的右下平台 (N-1, N-1)?

什么时候可以联通,答案是一个单调函数,就可以用二分查找来做,找到最少耗时。

class Solution {
    private int[] di = {1,-1,0,0};
    private int[] dj = {0,0,-1,1};
    public int swimInWater(int[][] grid) {
        int N = grid.length;
        if(N<=0) return -1;

        int low = 0;
        int high = N*N;

        while(low<high){
            int mid = (low+high) / 2;
            if(possible(mid, grid)){
                high = mid;
            }else{
                low = mid+1;
            }
        }

        return low;
    }

    private boolean possible(int T, int[][] grid){

        int N = grid.length;
        Stack<Integer> stack = new Stack<Integer>();
        Set<Integer> seen = new HashSet<Integer>();

        stack.push(0);
        seen.add(0);
        if(grid[0][0]>T) return false;

        while(!stack.empty()){
            int z = stack.pop();
            int i = z/N, j = z%N;
            if(i==N-1 && j==N-1)
                return true;
            
            for(int k = 0; k < 4;k++ ){
                int ii = i+di[k], jj = j+dj[k];
                int next = ii*N + jj;
                if(ii>=0 && ii<N && jj>=0 && jj<N && !seen.contains(next) && grid[ii][jj]<=T){
                    seen.add(next);
                    stack.push(next);
                }
            }
        }
        return false;
    }
}

33. 搜索旋转排序数组

题目链接

一个升序排列的整数数组 nums ,和一个整数 target 。

假设按照升序排序的数组在预先未知的某个点上进行了旋转。(例如,数组 [0,1,2,4,5,6,7] 可能变为 [4,5,6,7,0,1,2] )。

请你在数组中搜索 target ,如果数组中存在这个目标值,则返回它的索引,否则返回 -1。

条件: nums 中的每个值都 独一无二 nums 肯定会在某个点上旋转

class Solution {
    public int search(int[] nums, int target) {
        int length = nums.length;
        
        if(length<1)
            return -1;
        if(length==1) {
            return target == nums[0] ? 0 : -1;
        }

        int left = 0, right = length - 1;
        while(left <= right){
            int mid = (left + right) / 2;
            if(nums[mid]==target)
                return mid;
            
            if(nums[0] <= nums[mid]){
                if(nums[0] <= target && nums[mid] > target){
                    right = mid-1;
                }else{
                    left = mid+1;
                }
            }else{
                if(nums[mid]< target && target <= nums[right]){
                    left = mid+1;
                }else{
                    right = mid-1;
                }
            }
        }
        return -1;
    }
}

这里的二分查找是一种特殊情况,在[left,right]的范围中只有一个答案满足要求,设答案为ans。

while(left<=right){
    mid = (left+right)/2;
    p = func(mid);
    if(condition(p)){
        //找答案
        return mid;
    }else if(mid大了){
        right=mid-1;
    }else if(mid小了){
        left=mid+1;
    }
}

34. 在排序数组中查找元素的第一个和最后一个位置

题目链接

O(log n) 级别,二分查找

方法一

class Solution {
    // returns leftmost (or rightmost) index at which `target` should be
    // inserted in sorted array `nums` via binary search.
    private int extremeInsertionIndex(int[] nums, int target, boolean left) {
        int lo = 0;
        int hi = nums.length;

        while (lo < hi) {
            int mid = (lo + hi) / 2;
            if (nums[mid] > target || (left && target == nums[mid])) {
                hi = mid;
            }
            else {
                lo = mid+1;
            }
        }

        return lo;
    }

    public int[] searchRange(int[] nums, int target) {
        int[] targetRange = {-1, -1};

        int leftIdx = extremeInsertionIndex(nums, target, true);

        // assert that `leftIdx` is within the array bounds and that `target`
        // is actually in `nums`.
        if (leftIdx == nums.length || nums[leftIdx] != target) {
            return targetRange;
        }

        targetRange[0] = leftIdx;
        targetRange[1] = extremeInsertionIndex(nums, target, false)-1;

        return targetRange;
    }
}

方法二,两次二分查找,分别使用(L+R)/2(L+R+1)/2

class Solution {
    public int[] searchRange(int[] nums, int target) {
        // 在数组中的开始位置 和 结束位置
        // 时间复杂度O(log n)
        
        // 1 找左边
        int[] ans = new int[2];
        int n = nums.length;
        if(n==0) return new int[]{-1,-1};
        if(n==1){
            if(nums[0]==target) return new int[]{0,0};
            return new int[]{-1,-1};
        }
        int L = 0;
        int R = n-1;
        while(L < R){
            int mid = (L + R) / 2;
            if(nums[mid] >= target){
                R = mid;
            }else{
                L = mid+1;
            }
        }
        if(nums[L]!=target) return new int[]{-1,-1};
        ans[0] = L;
        R = n-1;
        while(L < R){
            int mid = (L+R+1)/2;
            if(nums[mid] <= target){
                L = mid;
            }else{
                R = mid-1;
            }
        }
        ans[1] = R;
        return ans;
    }
}

数组

31. 下一个排列

题目链接

class Solution {
    public void nextPermutation(int[] nums) {
        int length = nums.length;
        
        for(int i=length-2;i>=0; i--){
            if(nums[i+1]>nums[i]){
                int max = i+1;
                for(int j=i+2;j<length;j++){
                    if(nums[j]>nums[i] && nums[j]<nums[max]){
                        max=j;
                    }
                }
                swap(nums, max, i);
                Arrays.sort(nums,i+1,length);
                return;
            }
        }
        Arrays.sort(nums);
        return;
    }
    
    private void swap(int[] nums, int i, int j){
        int tmp = nums[i];
        nums[i] = nums[j];
        nums[j] = tmp;
    }
}

169. 多数元素

链接

分治

class Solution {
    public int majorityElement(int[] nums) {
        // 
        int n = nums.length;
        return fenzhi(nums, 0, n-1);
    }

    private int fenzhi(int[] nums, int left, int right){
        if(left == right){
            return nums[left];
        }

        int mid = (left+right) / 2;
        int leftM = fenzhi(nums, left, mid);
        int rightM = fenzhi(nums, mid+1, right);

        int leftCount = count(nums, leftM, left, right);
        int rightCount = count(nums, rightM, left,right);

        return leftCount > rightCount ? leftM : rightM;

    }

    private int count(int [] nums, int target, int left, int right){
        int ans = 0;
        for(int i= left; i <= right; i++)
            if(nums[i]==target)
                ans++;
        return ans;
    }
}

Boyer-Moore 投票算法

class Solution {
    public int majorityElement(int[] nums) {
        // 
        int count = 0;
        int target = 0;
        for(int num: nums){
            if(count == 0){
                target = num;
            }
            count += (target == num) ? 1 : -1;
        }
        return target;
    }
}

数学

279. 完全平方数

链接


搜索、回溯

N皇后

题目链接

搜索回溯,从左上到右下这个方向的同一条斜线上,行下标列下标相同;从右上到左下这个方向,行下标列下标相同。

通过$\left(2^{n}-1\right) \&\left(\sim\left(\text {columns} \mid \text {diagonals}{1} \mid \text {diagonals}{2}\right)\right)$获得这一行合法的位置。其中$2^{n}$用$1«n$得到

有两个重要位运算性质:

  • $x \&(-x)$可以获得$x$的二进制表示中最低位1的位置
  • $x \&(x-1)$将$x$的二进制表示中的最低位的 1 置成 0

position二进制的10,表示尝试下标为1的列. queens[row]=1; 100为下标为2的列queens[row]=2;用下面位运算

  • 下标等于:Integer.bitCount(position-1)
class Solution {
    public List<List<String>> solveNQueens(int n) {
        int[] queens = new int[n];
        Arrays.fill(queens, -1);
        List<List<String> > solutions = new ArrayList<List<String>>();
        solve(solutions, queens, n, 0, 0, 0, 0);
        return solutions;
    }

    // n, 当前搜索行数,当前列标志,当前 \ 方向斜线标志,当前 / 方向斜线标志。
    private void solve(List<List<String>> solutions, int[] queens, int n, int row, int columns, int diagonals1, int diagonals2 ) {
        if(row==n){
            List<String> board = generateBoard(queens, n);
            solutions.add(board);
            return;
        }
        int avaliablePosition = ((1<<n) -1) & (~(columns | diagonals1 | diagonals2));
        while(avaliablePosition!=0){
            int position = avaliablePosition & (-avaliablePosition); // 比如二进制的10,表示尝试下标为1的列. queens[row]=1; 100为下标为2的列。
            avaliablePosition = avaliablePosition & (avaliablePosition - 1);
            //计算占用哪一列
            int column = Integer.bitCount(position-1);
            queens[row] = column;
            solve(solutions, queens, n, row+1, columns | position, (diagonals1 | position) << 1, (diagonals2 | position) >> 1);
            queens[row] = -1; //回溯
        }
    }

    private List<String> generateBoard(int[] queens, int n){
        // queens放的是每一行,皇后所在列的下标
        List<String> board = new ArrayList<String>();
        for(int i=0; i<n ; i++){
            char[] row = new char[n];
            Arrays.fill(row , '.');
            row[queens[i]] = 'Q';
            board.add(new String(row));
        }
        return board;
    }
}

N皇后II

题目链接

返回N皇后问题的不同解决方案数量。同样的方法,solve函数返回int,累加起来。

class Solution {
    public int totalNQueens(int n) {
        return solve(n, 0, 0, 0, 0);
    }

    public int solve(int n, int row, int columns, int diagonals1, int diagonals2) {
        if (row == n) {
            return 1;
        } else {
            int count = 0;
            int availablePositions = ((1 << n) - 1) & (~(columns | diagonals1 | diagonals2));
            while (availablePositions != 0) {
                int position = availablePositions & (-availablePositions);
                availablePositions = availablePositions & (availablePositions - 1);
                count += solve(n, row + 1, columns | position, (diagonals1 | position) << 1, (diagonals2 | position) >> 1);
            }
            return count;
        }
    }
}

139. 单词拆分

题目链接

给定一个非空字符串 s 和一个包含非空单词列表的字典 wordDict,判定 s 是否可以被空格拆分为一个或多个在字典中出现的单词。

  • 暴力,超时
    class Solution {
      public boolean wordBreak(String s, List<String> wordDict) {
          if("".equals(s))
              return true;
          for(int i = 0; i < s.length(); i++){
              String str = s.substring(0,i+1);
              if(wordDict.contains(str) && wordBreak(s.substring(i+1), wordDict))
                  return true;
          }
    
          return false;
      }
    }
    
  • 回溯+记忆化,一个Boolean数组保存后缀是否可以匹配
    class Solution {
      public boolean wordBreak(String s, List<String> wordDict) {
          return help(s, new HashSet(wordDict), new Boolean[s.length()], 0);
      }
    
      public boolean help(String s, List<String> wordDict, Boolean [] memo, int start){
            
          if(start==s.length())
              return true;
    
          if(memo[start]!=null)//true和false都保存
              return memo[start];
    
          for(int i = start; i < s.length(); i++){
              String str = s.substring(start,i+1);
              if(wordDict.contains(str) && help(s, wordDict, memo, i+1)){
                  return memo[start]=true;
              }
          }
    
          return memo[start] = false;   
      }
    }
    
class Solution {
    public boolean wordBreak(String s, List<String> wordDict) {
        if("".equals(s))
            return true;
        
        int length = s.length();
        HashSet<String> dict = new HashSet<>(wordDict);
        Boolean[] memo = new Boolean[length]; //要用 大写开头的 Boolean

        return wordBreak(s, 0, length, dict, memo);
    }

    private boolean wordBreak(String s, int startIndex, int length, HashSet<String> dict, Boolean[] memo){
        if(startIndex==length)
            return true;
        
        if(memo[startIndex]!=null)
            return memo[startIndex];

        for(int i=startIndex; i<length; i++){
            String substr = s.substring(startIndex, i+1);
            if(dict.contains(substr)){
                if(wordBreak(s, i+1, length, dict,memo)){
                    return memo[startIndex] = true;
                }
            }
        }
        
        return memo[startIndex] = false;
    }
}

Leetcode 37 数独

题目链接

像游戏的题目,9*9的格子。 输入9行,每行为空格隔开的9个数字,为0的地方就是需要填充的。 要求每行每列数字各不相同,并且每个小九宫格里面的数字也各不相同。典型的方法是回溯法,根据某一个格子里填入的数字,继续搜索下一个可以填入的数字

参考:[编程题 解数独问题 (Leetcode 37题)](https://blog.csdn.net/weixin_43795395/article/details/105404358)

DFS,成功了返回True,不继续往下面搜索,也跟着返回true;如果搜索失败了,回溯,继续搜索。 下面的方法用3个成员,记录了每行,每列,每个小方格已经使用的数字数量。利用bitset.count()bitset.test()bitset.size()来处理。

c++

class Solution {
public:
    //得到所有能填的数字
    bitset<9> getPossibleStatus(int x, int y)
    {
        return ~(rows[x] | cols[y] | cells[x/3][y/3]);
    }

    //使用 getNext() 选择能填的数字最少的格子开始填,这样填错的概率最小,回溯次数也会变少
    vector<int> getNext(vector<vector<char>>& board)
    {
        vector<int> ret;
        int minCnt = 10;
        for(int i=0; i<9; i++)
            for(int j=0; j < 9; j++)
            {
                if(board[i][j]!='.')continue;
                auto remain = getPossibleStatus(i,j);
                if(remain.count()>=minCnt) continue;
                ret = {i, j}; // ret表示位置
                minCnt = remain.count();
            }
        
        return ret;
    }
    
    // 在填入和回溯时负责更新存储信息
    void fillNum(int x, int y, int n, bool fillFlag)
    {
        rows[x][n] = fillFlag ? 1 : 0;
        cols[y][n] = fillFlag ? 1 : 0;
        cells[x/3][y/3][n] = fillFlag ? 1 : 0;
    }

    bool dfs(vector<vector<char>>& board, int cnt)
    {
        if(cnt==0) return true;

        auto next = getNext(board);
        auto bits = getPossibleStatus(next[0], next[1]);
        for(int i=0; i<bits.size(); i++){
            if(!bits.test(i)) continue;
            board[next[0]][next[1]] = '1'+i;
            fillNum(next[0],next[1],i, true);
            if(dfs(board, cnt-1)){
                return true;
            }
            fillNum(next[0],next[1],i, false);
            board[next[0]][next[1]] = '.';
        }
        return false;

    }
    void solveSudoku(vector<vector<char>>& board) {
        // board
        rows = vector<bitset<9>>(9, bitset<9>());
        cols = vector<bitset<9>>(9, bitset<9>());
        cells = vector<vector<bitset<9>> >(3,vector<bitset<9>>(3, bitset<9>()));

        int cnt = 0;
        
        for (int i=0; i < 9; i++)
        {
            for(int j=0; j < 9; j++){
                if(board[i][j]=='.') {
                    cnt++;
                    continue;
                }
                int n = board[i][j] - '1';
                rows[i] |= (1<<n);
                cols[j] |= (1<<n);
                cells[i/3][j/3] |= (1<<n);
            }
        }
        dfs(board, cnt);
    }
private:
    vector<bitset<9>> rows; // 9行 
    vector<bitset<9>> cols; // 9列
    // 小九宫格 2维,3*3
    vector<vector<bitset<9>> > cells;
};

93 复原IP地址

题目链接 需要找出所有方案,进行DFS 回溯搜索。使用Stringbuilder,可以将当前查找的数字用int记录(不需要用string。还可以方便判断是否超过0~255的范围)

class Solution {
    private int[] segments;
    private List<String> result;

    public List<String> restoreIpAddresses(String s) {
        int n = s.length();
        
        if (n < 4 || n > 12) {
            return new ArrayList<>();
        }

        char[] chars = s.toCharArray();
        segments = new int[4];
        result = new ArrayList<>();
        // dfs搜索,回溯剪枝条
        dfs(s, 0, 0);
        return result;
    }

    private void dfs(String s, int curSeg, int startIndex) {
        if(curSeg == 4 && startIndex == s.length()) {
            // 完成一个了
            StringBuilder sb = new StringBuilder();
            for(int i=0; i<4; i++) {
                sb.append(segments[i]);
                if (i!=3) {
                    sb.append('.');
                }
            }
            result.add(sb.toString());
            return;
        }

        if (curSeg == 4 || startIndex == s.length()) {
            return;
        }

        // 第一个字符是 ‘0’,不能当作前导0
        if (s.charAt(startIndex)=='0') {
            segments[curSeg] = 0;
            dfs(s, curSeg + 1, startIndex + 1);
            return; // 可以return了
        }

        int curSegNumber = 0;
        for(int i = startIndex; i<startIndex+3 && i<s.length() ; i++) {
            int num = (s.charAt(i)-'0');
            if (curSegNumber < 255 / 10 || (curSegNumber == 255 / 10 && num <= 5)){  //这个逻辑应该用自 超过 int32 的情况下,这里不用这么小心翼翼。
                curSegNumber = curSegNumber * 10 + num;
                segments[curSeg] = curSegNumber;
                dfs(s, curSeg + 1, i + 1);
            } else {
                break;
            }
        }
    }
}
  • 动态规划 状态转移方程:
\[d p[i]=d p[j] \& \& \operatorname{check}(s[j \ldots i-1])\]
class Solution {
    public boolean wordBreak(String s, List<String> wordDict) {
        int length = s.length();
        if(length==0)
            return true;
        HashSet<String> dict = new HashSet<>(wordDict);
        boolean[] dp = new boolean[length+1];
        dp[0]=true;
        for(int i=1; i<=length; i++){
            for(int j=0;j<i;j++){
                if(dp[j]==true && dict.contains(s.substring(j,i))){
                    dp[i]=true;
                    break;
                }
            }
        }
        return dp[length];
    }
}

17. 电话号码的字母组合

用两个辅助数组,方便打印选择字符

class Solution {
    int[] tint = {0,0,2,5,8,11,14,18,21,25};
    int[] count = {0,0,3,3,3,3,3,4,3,4};
    public List<String> letterCombinations(String digits) {
        //2~9
        List<String> ans = new ArrayList<>();
        int length = digits.length();
        if(length==0)  return ans;
        char[] chars = new char[length];
        
        dfs(digits, 0, length, ans, chars);

        return ans;

    }

    private void dfs(String digits, int cur, int length, List<String> ans, char[] chars){
        if(cur == length){
            ans.add(new String(chars));
            return;
        }
        int num = digits.charAt(cur) - '0';
        int start = tint[num] - count[num] + 1;
        int end = tint[num];
        for(int i=start; i<= end; i++){
            chars[cur] = (char)('a' + i);
            dfs(digits, cur+1, length, ans, chars);
        }
    }
}

39. 组合总和

题目链接

数字可以重复使用多次。

class Solution {
    public List<List<Integer>> combinationSum(int[] candidates, int target) {
        List<List<Integer> > res = new ArrayList<List<Integer>>();
        List<Integer> path = new ArrayList<>();

        int n = candidates.length;
        if( n == 0 ){
            return res;
        }

        search(candidates, res, path, 0, n, target);
        return res;
    }
    private void search(int[] candidates, List<List<Integer> >res, List<Integer> path, int idx, int n, int target){
        
        if(idx == n){
            return;
        }

        if(target == 0){
            res.add(new ArrayList<Integer>(path));
            return;
        }

        // 不要第idx个
        search(candidates, res, path, idx+1, n, target);
        if(target >= candidates[idx]){
            path.add(candidates[idx]);
            search(candidates, res, path, idx, n, target - candidates[idx]);
            path.remove(path.size()-1);
        }
    }
}

对candidates排序,以减少回溯次数。

class Solution {

    private List<List<Integer>> res = new ArrayList<>();

    public List<List<Integer>> combinationSum(int[] candidates, int target) {
        List<Integer> path = new ArrayList<>();
        Arrays.sort(candidates);
        backtrack(path,candidates,target,0,0);
        return res;
    }

    private void backtrack(List<Integer> path,int[] candidates,int target,int sum,int begin) {
        if(sum == target) {
            res.add(new ArrayList<>(path));
            return;
        }
        for(int i = begin;i < candidates.length;i++) {
            int rs = candidates[i] + sum;
            if(rs <= target) {
                path.add(candidates[i]);
                backtrack(path,candidates,target,rs,i);
                path.remove(path.size()-1);
            } else {
                break;
            }
        }
    }
}

394. 字符串解码

链接

class Solution {
    int ptr;
    public String decodeString(String s) {

        int n = s.length();
        Stack<String> stack = new Stack<>();
        ptr = 0;
        while(ptr<n){
            char c = s.charAt(ptr);
            if(Character.isDigit(c)){
                String digits = getDigits(s);
                stack.push(digits);
            } else if(Character.isLetter(c) || c == '['){
                stack.push(String.valueOf(c));
                ptr++;
            } else {
                // ']'
                ++ptr;
                Stack<String> sub = new Stack<String>();
                while(!"[".equals(stack.peek())){
                    sub.push(stack.pop());
                }
                Collections.reverse(sub);
                // 左括号出栈
                stack.pop();
                //之后是个数字
                int repeat = Integer.parseInt(stack.pop());
                String o = decodeString(sub);
                while((repeat--)>0){
                    stack.push(o);
                }
            }
        }
        return decodeString(stack);
    }

    private String getDigits(String s){
        StringBuilder sb = new StringBuilder();
        while (Character.isDigit(s.charAt(ptr))){
            sb.append(s.charAt(ptr));
            ptr++;
        }
        return sb.toString();
    }

    private String decodeString(List<String> v){
        StringBuilder sb = new StringBuilder();
        for(String s : v){
            sb.append(s);
        }
        return sb.toString();
    }
}

339 除法求值

题目链接

思路:链接

class Solution {
    static Map<String, String> parents;
    static Map<String, Double> val;


    public String find(String x) {

        if (!x.equals(parents.get(x))) {
            String tmpParent = parents.get(x);
            String root = find(tmpParent);
            double oldVal = val.get(x);
            val.put(x, oldVal * val.get(tmpParent));
            parents.put(x, root);
        }
        return parents.get(x);
    }

    public double[] calcEquation(List<List<String>> equations, double[] values, List<List<String>> queries) {
        parents = new HashMap<>();
        val = new HashMap<>();
        int i = 0;
        for (List<String> equation : equations) {
            String from = equation.get(0);
            String to = equation.get(1);
            double cur = values[i];
            if (!parents.containsKey(from) && !parents.containsKey(to)) {
                parents.put(from, to);
                val.put(from, cur);
                parents.put(to, to);
                val.put(to, 1.0);
            } else if (!parents.containsKey(from)) {
                parents.put(from, to);
                val.put(from, cur);
            } else if (!parents.containsKey(to)) {
                parents.put(to, from);
                val.put(to, 1 / cur);
            } else {
                String pa = find(from);
                String pb = find(to);
                if (!pa.equals(pb)) {
                    parents.put(pa, pb);
                    val.put(pa, cur * val.get(to) / val.get(from));
                }
            }
            i++;
        }
        i = 0;
        double[] res = new double[queries.size()];
        for (List<String> query : queries) {
            String from = query.get(0);
            String to = query.get(1);
            if (!parents.containsKey(from) || !parents.containsKey(to)) {
                res[i++] = -1;
                continue;
            }
            String pa = find(from);
            String pb = find(to);
            if (!pa.equals(pb)) res[i] = -1;
            else {
                res[i] = val.get(from) / val.get(to);
            }
            i++;
        }
        return res;
    }
}

90. 子集 II

链接

数组有重复数字,列举所有子集。 先排序,:[1, 2, 2, 2] 如果上一个位置没有采纳,并且上一个位置的数字和当前数字相同,则return剪枝。

class Solution {
    private int n;
    private List<List<Integer>> ans;
    
    public List<List<Integer>> subsetsWithDup(int[] nums) {
        Arrays.sort(nums);
        n = nums.length;
        ans = new ArrayList<List<Integer> >();
        List<Integer> set = new ArrayList<Integer>();
        dfs(false, 0, nums, set);
        return ans;
    }

    private void dfs(boolean useLast, int cur, int[] nums, List<Integer> set) {
        if( cur==n ){
            ans.add(new ArrayList<Integer>(set));
            return;
        }

        // 不要当前
        dfs(false, cur+1, nums, set);
        
        if(!useLast && cur>0 && nums[cur-1]==nums[cur])
            return;
        
        set.add(nums[cur]);
        dfs(true, cur+1, nums, set);
        set.remove(set.size()-1);
    }
}

链表

反转链表

class Solution {

    public ListNode reverseList(ListNode head) {
        
        ListNode prev=null;
        ListNode cur=head;

        while(cur!=null){
            ListNode next = cur.next;
            cur.next=prev;
            prev=cur;
            cur=next;
        }
        return prev;
    }
};

反转部分链表

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode(int x) { val = x; }
 * }
 */
 public class Solution {
        public ListNode reverseBetween(ListNode head, int m, int n) {
            ListNode dummy = new ListNode(0);
            dummy.next = head;
            ListNode prev = dummy;
            while (m > 1) {
                prev = prev.next;
                m--;
                n--;
            }
            head = prev.next;
            while (n > 1) {
                ListNode next = head.next;
                head.next = head.next.next;
                next.next = prev.next;
                prev.next = next;
                n--;
            }
            return dummy.next;
        }
}

 public class Solution {
        public ListNode reverseBetween(ListNode head, int m, int n) {

            ListNode con=null;
            ListNode prev=null;
            ListNode cur=null;
            ListNode next=null;
            ListNode tail=null;
            cur=head;
            while(m>1){
                prev=cur;
                cur=cur.next;
                m--;n--;
            }
            con=prev;
            tail=cur;
            while(n>0){
                next=cur.next;
                cur.next=prev;
                prev=cur;
                cur=next;
                n--;
            }
            tail.next=cur;
            if(con!=null){
                con.next=prev;
                return head;
            }
            return prev;
        }
}

143. 重排链表

题目链接

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
class Solution {
    public void reorderList(ListNode head) {
        if(head==null || head.next==null){
            return;
        }

        ListNode ptr1=head;
        ListNode ptr2=head;

        //这种方式找到的 ptr1是中间节点或者 中间靠右的节点 mid=(left+right+1)/2,找到的节点就是最后结果中的尾部
        while(ptr2!=null && ptr2.next!=null){
            ptr1=ptr1.next;
            ptr2=ptr2.next.next;
        }
        ListNode last = ptr1;
        //把last节点后面的全部压入栈里面
        Stack<ListNode> stack = new Stack<>();
        ptr1=ptr1.next;
        while(ptr1!=null){
            stack.push(ptr1);
            ptr1=ptr1.next;
        }
        ptr1=head;
        while(!stack.empty()){
            ptr2=stack.pop();
            ptr2.next=ptr1.next;
            ptr1.next=ptr2;
            ptr1=ptr2.next;
        }
        last.next=null;
    }
}

字符串

面试题 17.13. 恢复空格

题目链接

以前写的题解

把文章断开,要求未识别的字符最少,返回最少的未识别字符数。

这里Tire数是从单词结尾往单词头部倒着连接的。

private Class Trie{
    public Trie[] next;
    boolean isEnd;
    public Trie(){
        next = new Trie[26];
        isEnd = false;
    }

    public void insert(String sentence){
        char[] chars = sentence.toCharArray();
        int n = chars.length;
        Trie cur = this;
        for(int i=n-1; i>=0; i--){
            int t = chars[i]-'a';
            if(cur.next[t]==null){
                cur.next[t] = new Trie();
            }
            cur = cur.next[t];
        }
        cur.isEnd = true;
    }
}

动态规划

class Solution {
    private class Trie{
        public Trie[] next;
        boolean isEnd;
        public Trie(){
            next = new Trie[26];
            isEnd = false;
        }

        public void insert(String sentence){
            char[] chars = sentence.toCharArray();
            int n = chars.length;
            Trie cur = this;
            for(int i=n-1; i>=0; i--){
                char c = chars[i];
                if(cur.next[c-'a']==null){
                    cur.next[c-'a'] = new Trie();
                }
                cur = cur.next[c-'a'];
            }
            cur.isEnd = true;
        }

    }
    public int respace(String[] dictionary, String sentence) {
        // 尽量让未匹配的字符数最少
        Trie trieTree = new Trie();
        for(String s: dictionary){
            trieTree.insert(s);
        }
        int n = sentence.length();
        char[] chars = sentence.toCharArray();
        int[] dp = new int[n+1];
        //动态规划
        Arrays.fill(dp, Integer.MAX_VALUE);
        dp[0]=0;
        for(int i=1;i<=n;i++){
            dp[i] = dp[i-1]+1;
            Trie cur = trieTree;
            for(int j=i;j>=1;j--){
                int t = chars[j-1]-'a';
                if(cur.next[t]==null) break;
                else if(cur.next[t].isEnd){
                    dp[i] = Math.min(dp[i],dp[j-1]);
                }
                cur = cur.next[t];
            }
        }
        return dp[n];
    }
}

字符串

Z字形变换

Z字形变换

class Solution {

    public String convert(String s, int numRows) {

        if (numRows == 1) return s;

        StringBuilder sb = new StringBuilder();
        int n = s.length();
        int period = 2*numRows - 2;
        for(int i=0;i<numRows;i++){
            for( int j=0; j+i<n;j+=period ){
                sb.append(s.charAt(j+i));
                if(i!=0 && i!=numRows-1 && j+period-i<n)
                    sb.append(s.charAt(j+period-i));
            }
        }

        return sb.toString();
    }
}

概率题目

比如蓄水池算法相关题目

398. 随机数索引

题目链接

概率相同,蓄水池抽样算法。 扫描,然后以1, 1/2, 1/3, 1/4…的概率保留当前扫描到的数字。

class Solution {
    private int[] nums;

    public Solution(int[] nums) {
        this.nums = nums;
    }
    
    public int pick(int target) {
        Random r = new Random();
        int index=1;
        int ans=0;
        int n = nums.length;
        for(int i=0;i<n;i++){
            if(nums[i]==target){
                int j=r.nextInt() % index;
                if(j==0){
                    ans=i;
                }
                index++;
            }
        }
        return ans;
    }
}

382. 链表随机节点

题目链接

给定一个单链表,随机选择链表的一个节点,并返回相应的节点值。保证每个节点被选的概率一样。

class Solution {
    private ListNode head;
    /** @param head The linked list's head.
        Note that the head is guaranteed to be not null, so it contains at least one node. */
    public Solution(ListNode head) {
        this.head = head;
    }
    
    /** Returns a random node's value. */
    public int getRandom() {
        Random r = new Random();
        int ans=0;
        ListNode node = head;
        int i=1;
        while(node!=null){
            int j = r.nextInt()%i;
            if(j==0){
                ans = node.val;
            }
            node=node.next;
            i++;
        }
        return ans;
    }
}

结构设计

LRU

用双向链遍,集合,链表中的元素包含<key, value>。包含key是为了,在容量满的时候添加一个元素,删除链表最后的那个数据,能哥获取到key,删除map中的记录。

c++ 可以直接用STL实现

class LRUCache {
private:
    list<pair<int, int> > cache;
    //
    unorder_map<int, list<pair<int, int>>::iterator> map;
    int capacity;
public:
    LRUCache(int capacity) {
        this->capacity = capacity;
    }
    
    int get(int key) {
        //get
        if(map.find(key)!=map.end()){
            pair<int,int> p = *map[key];
            int val = p.second;
            cache.erase(map[key]);
            cache.push_front(make_pair(key, val));
            map[key]=cache.begin();
            return p.second;
        }
        return -1;
    }
    
    void put(int key, int value) {
        //put
        if(map.find(key)!=map.end()){
            pair<int,int> p = *map[key];
            list.erase(map[key]);
            list.push_front(make_pair(key,value));
            map[key]=list.begin();
        }else{
            if(list.size()>=this->count){
                pair<int,int >p = list.back();
                int lastKey = p.first;
                list.pop_back();
                map.erase(lastKey);
            }
            list.push_front(make_pair(key, value));
            map[key]=list.begin();
        }
    }
};

java稍微写的代码多一点。

public class LRUCache {
    private int capacity;
    private HashMap<Integer, Node> map = new HashMap<Integer, Node>();
    private LinkedList<Node> list = new LinkedList<Node>();

    public LRUCache(int capacity) {
       this.capacity = capacity;
    }

    /**
     * 如果存在,把当前结点移动到双向链表的头部
     */
    public int get(int key) {
        if(map.get(key) != null) {
            // map找不到
            Node node = map.get(key);
            list.remove(node);
            list.addFirst(node);
            map.put(key, node);
            return node.value;
        }
        return -1;
    }

    /**
     * 如果哈希表的容量满了,就要删除一个链表末尾元素,然后在链表头部插入新元素
     *
     * @param key
     * @param value
     */
    public void put(int key, int value) {
        if(map.get(key)!=null) {
            Node node = map.get(key);
            list.remove(node);
            node.value=value;
            list.addFirst(node);
            map.put(key,node);
        }else{
            if(list.size()==this.capacity) {
                Node node = list.getLast();
                list.removeLast();
                map.remove(node.key);
            }
            Node newNode = new Node(key,value);
            list.addFirst(newNode);
            map.put(key, newNode);
        }
    }

   
}

class Node{
    int key;
    int value;
    Node(int key, int value){
        this.key = key;
        this.value = value;
    }
}

Trie 前缀树

实现前缀树

全部都是小写。就用一个26位的数组,代表小写字母的’a’-‘z’。如果包含大写字母或者中文,可以该用字典数据结构HashMap<Character, Trie>的方法。

class Trie {
    private Trie[] children; 
    private boolean isEnd = false;
    /** Initialize your data structure here. */
    public Trie() {
        this.children = new Trie[26];
        this.isEnd=false;
    }
    
    /** Inserts a word into the trie. */
    public void insert(String word) {
        char[] chars = word.toCharArray();
        Trie node = this;
        int n = chars.length;
        for(int i=0;i<n;i++){
            int t = chars[i]-'a';
            if(node.children[t]==null){
                node.children[t]=new Trie();
            }
            node = node.children[t];
        }
        node.isEnd=true;
    }
    
    /** Returns if the word is in the trie. */
    public boolean search(String word) {
        char[] chars = word.toCharArray();
        int n = chars.length;
        Trie node = this;
        for(int i=0;i<n;i++){
            int t = chars[i]-'a';
            if(node.children[t]==null){
                return false;
            }
            node = node.children[t];
        }
        return node.isEnd;
    }
    
    /** Returns true if there is any word in the trie that starts with the given prefix. */
    public boolean startsWith(String prefix) {
        char[] chars = prefix.toCharArray();
        int n = prefix.length();
        Trie node = this;
        for(int i=0;i<n;i++){
            int t = chars[i]-'a';
            if(node.children[t]==null){
                return false;
            }
            node = node.children[t];
        }
        return true;
    }
}
/**
 * Your Trie object will be instantiated and called as such:
 * Trie obj = new Trie();
 * obj.insert(word);
 * boolean param_2 = obj.search(word);
 * boolean param_3 = obj.startsWith(prefix);
 */

LFU

最不经常最少使用 LFUCache。 定义两个哈希表,第一个Map<key,node>。第二个Map<freq, list>

class LFUCache {
    int minfreq, capacity;
    Map<Integer, Node> key_table;
    Map<Integer, LinkedList<Node>> freq_table;

    public LFUCache(int capacity) {
        this.minfreq = 0;
        this.capacity = capacity;
        key_table = new HashMap<Integer, Node>();
        freq_table = new HashMap<Integer, LinkedList<Node>>();
    }

    public int get(int key) {
        // 提前预判容量
        if (capacity == 0) {
            return -1;
        }

        // 没找到
        if (!key_table.containsKey(key)) {
            return -1;
        }

        // 增加访问次数,根据情况看是否需要更新minfreq
        Node node = key_table.get(key);
        int val = node.val, freq = node.freq;

        freq_table.get(freq).remove(node);
        if(freq_table.get(freq).size()==0) {
            freq_table.remove(freq);
            if(minfreq == freq) 
                minfreq++;
        }

        // 在频率为`freq+1`对应的双向链表中加入更新后的节点信息
        LinkedList<Node> list = freq_table.getOrDefault(freq+1, new LinkedList<Node>());
        node = new Node(key, val, freq+1);
        list.offerFirst(node);
        freq_table.put(freq+1, list);
        key_table.put(key, node);
        return val;
    }

    public void put(int key, int val) {
        if (capacity==0) {
            return;
        }
        int freq = 1;
        if (!key_table.containsKey(key)) {
            if (key_table.size() == this.capacity) {
                Node node = freq_table.get(minfreq).peekLast();
                key_table.remove(node.key);
                freq_table.get(minfreq).pollLast();
                if(freq_table.get(minfreq).size()==0) {
                    freq_table.remove(minfreq);
                }
            }

            minfreq = 1;
        } else {
            Node node = key_table.get(key);
            int prevFreq = node.freq;
            freq = prevFreq+1;
            freq_table.get(prevFreq).remove(node);
            if(freq_table.get(prevFreq).size()==0) {
                freq_table.remove(prevFreq);
                if (minfreq==prevFreq) {
                    minfreq++;
                }
            }
        }

        LinkedList<Node> list = freq_table.getOrDefault(freq, new LinkedList<Node>());
        list.offerFirst(new Node(key, val, freq));
        freq_table.put(freq, list);
        key_table.put(key, freq_table.get(freq).peekFirst());
    }
}

class Node {
    int key, val, freq;

    Node(int key, int val, int freq) {
        this.key = key;
        this.val = val;
        this.freq = freq;
    }
}

拓扑排序

课程表

题目链接

  • 深度优先拓扑排序,
class Solution {    

    private List<List<Integer>> edges;
    private int[] visited; // 需要有3个状态

    public boolean canFinish(int numCourses, int[][] prerequisites) {

        // 深度优先遍历,需要转变为领接矩阵
        edges = new ArrayList<List<Integer>>();
        visited = new int[numCourses]; //初始化都是0
        for(int i=0; i < numCourses; i++){
            edges.add(new ArrayList<Integer>());
        }

        for(int[] info: prerequisites){
            edges.get(info[1]).add(info[0]);
        }

        for(int i=0; i<numCourses; i++){
            if(visited[i]==0){
                if(!dfs(i)){
                    return false;
                }
            }
        }
        return true;
        
    }

    private boolean dfs(int u){
        visited[u] = 1;
        
        for(int v : edges.get(u)){
            if(visited[v]==1)
                return false;
            else if(visited[v]==0){
                if(!dfs(v)){
                    return false;
                }
            }
        }
        visited[u] = 2;
        return true;
    }
}
  • 广度优秀

深度优先是一种【逆向思维】,最先进入栈里面的节点是排在拓扑排序最后面的节点。也可以正向思维,顺序地生成拓扑排序。

用一个队列存储现在入度为0的节点,访问一次就poll一次 队列 + while循环 + 邻接矩阵扫描

class Solution {    
    private List<List<Integer>> edges;
    private int[] indeg;
    public boolean canFinish(int numCourses, int[][] prerequisites) {

        // 深度优先遍历,需要转变为领接矩阵
        edges = new ArrayList<List<Integer>>();
        indeg = new int[numCourses];

        for(int i=0; i < numCourses; i++){
            edges.add(new ArrayList<Integer>());
        }

        for(int[] info: prerequisites){
            edges.get(info[1]).add(info[0]);
            indeg[info[0]]++;
        }
        
        Queue<Integer> queue = new LinkedList<>();

        for(int i=0; i< numCourses; i++){
            if(indeg[i]==0)
                queue.offer(i); //入度为0的全部放进去
        }

        int visited = 0;
        while(!queue.isEmpty()){
            int u = queue.poll();
            visited++;
            for(int v : edges.get(u)){
                indeg[v]--;
                if(indeg[v]==0){
                    queue.offer(v);
                }
            }
        }

        return visited == numCourses;
    }
}

课程表II

题目链接

方法和上一题相同,可以多使用一些数据结构答案。

  • 广度优先
class Solution {
    // 存储有向图
    private List<List<Integer>> edges;
    // 存储每个节点的入度
    private int[] indeg;
    // 存储答案
    private int[] result;
    // 答案下标
    private int index;

    public int[] findOrder(int numCourses, int[][] prerequisites) {

        indeg = new int[numCourses];
        result = new int[numCourses];
        edges = new ArrayList<List<Integer>>();
        for(int i=0; i< numCourses; i++){
            edges.add(new ArrayList<Integer>());
        }

        for(int[] info : prerequisites){
            edges.get(info[1]).add(info[0]);
            indeg[info[0]]++;
        }

        Queue<Integer> queue = new LinkedList<Integer>();
        // 把所有入度为0的课程编号放进队列
        for(int i = 0; i < numCourses; i++){
            if(indeg[i]==0){
                queue.offer(i);
            }
        }
        index = 0;
        while(!queue.isEmpty()){
            int u = queue.poll();
            result[index++]=u;
            for(int v : edges.get(u)){
                --indeg[v];
                if(indeg[v]==0){
                    queue.offer(v);
                }
            }
        }
        if(index!=numCourses)
            return new int[0];
        return result;
    }
}
  • 深度优先
class Solution {
    // 存储有向图
    List<List<Integer>> edges;
    // 标记每个节点的状态:0=未搜索,1=搜索中,2=已完成
    int[] visited;
    // 用数组来模拟栈,下标 n-1 为栈底,0 为栈顶
    int[] result;
    // 判断有向图中是否有环
    boolean valid = true;
    // 栈下标
    int index;

    public int[] findOrder(int numCourses, int[][] prerequisites) {
        edges = new ArrayList<List<Integer>>();
        for (int i = 0; i < numCourses; ++i) {
            edges.add(new ArrayList<Integer>());
        }
        visited = new int[numCourses];
        result = new int[numCourses];
        index = numCourses - 1;
        for (int[] info : prerequisites) {
            edges.get(info[1]).add(info[0]);
        }
        // 每次挑选一个「未搜索」的节点,开始进行深度优先搜索
        for (int i = 0; i < numCourses && valid; ++i) {
            if (visited[i] == 0) {
                dfs(i);
            }
        }
        if (!valid) {
            return new int[0];
        }
        // 如果没有环,那么就有拓扑排序
        return result;
    }

    public void dfs(int u) {
        // 将节点标记为「搜索中」
        visited[u] = 1;
        // 搜索其相邻节点
        // 只要发现有环,立刻停止搜索
        for (int v: edges.get(u)) {
            // 如果「未搜索」那么搜索相邻节点
            if (visited[v] == 0) {
                dfs(v);
                if (!valid) {
                    return;
                }
            }
            // 如果「搜索中」说明找到了环
            else if (visited[v] == 1) {
                valid = false;
                return;
            }
        }
        // 将节点标记为「已完成」
        visited[u] = 2;
        // 将节点入栈
        result[index--] = u;
    }
}

数组

48. 旋转图像

给定一个 n × n 的二维矩阵表示一个图像。 将图像顺时针旋转 90 度。

方法1,两次矩阵操作, 转置 + 每行反转

class Solution {
    public void rotate(int[][] matrix) {
        int n = matrix.length;
        if(n==0)return;
        // 转置操作
        for(int i=0; i<n; i++)
            for(int j=i; j<n; j++){
                int tmp = matrix[j][i];
                matrix[j][i] = matrix[i][j];
                matrix[i][j] = tmp;
            }
        
        // 反转每一层
        for(int i=0; i<n; i++){
            int l = 0, r = n-1;
            while(l < r){
                int tmp = matrix[i][l];
                matrix[i][l] = matrix[i][r];
                matrix[i][r] = tmp;
                l++;r--;
            }
        }
    }
}

方法2,一次操作(略)

75. 颜色分类

题目链接 给定一个包含红色、白色和蓝色,一共 n 个元素的数组,原地对它们进行排序,使得相同颜色的元素相邻,并按照红色、白色、蓝色顺序排列。

双指针

class Solution {
    public void sortColors(int[] nums) {
        int n = nums.length;
        int red = -1, blue = n;
        for(int i=0; i<blue;i++){
            while(i < blue && nums[i]==2){
                blue--;
                swap(nums, blue, i);
            }
            if(nums[i]==0){
                red++;
                swap(nums, red, i);
            }
        }
    }

    private void swap(int[] nums, int i, int j){
        int tmp = nums[i];
        nums[i] = nums[j];
        nums[j] = tmp;
    }
}

字符串

49. 字母异位词分组

题目链接

给定一个字符串数组,将字母异位词组合在一起。字母异位词指字母相同,但排列不同的字符串。

方法1:排序数组分类

class Solution {
    public List<List<String>> groupAnagrams(String[] strs) {
        // 字母相同,排列不同
        int n = strs.length;
        Map<String, ArrayList> dict = new HashMap<>();
        for(int i=0; i<n; i++){
            char[] chars = strs[i].toCharArray();
            Arrays.sort(chars);
            String key = new String(chars);
            if(!dict.containsKey(key)) dict.put(key, new ArrayList());
            dict.get(key).add(strs[i]);
        }
        return new ArrayList(dict.values());
    }
}

方法2:按计数分类。字符计数(每个字符的出现次数)相同时。

class Solution {
    public List<List<String>> groupAnagrams(String[] strs) {
        if (strs.length == 0) return new ArrayList();
        Map<String, List> ans = new HashMap<String, List>();
        int[] count = new int[26];
        for (String s : strs) {
            Arrays.fill(count, 0);
            for (char c : s.toCharArray()) count[c - 'a']++;

            StringBuilder sb = new StringBuilder("");
            for (int i = 0; i < 26; i++) {
                sb.append('#');
                sb.append(count[i]);
            }
            String key = sb.toString();
            if (!ans.containsKey(key)) ans.put(key, new ArrayList());
            ans.get(key).add(s);
        }
        return new ArrayList(ans.values());
    }
}

4. 寻找两个正序数组的中位数

题目链接

给定两个大小分别为 mn 的正序(从小到大)数组 nums1nums2。请你找出并返回这两个正序数组的 中位数

中位数问题,转换为在两个有序数组里一起找第k小的数字。 主要思路:要找到第 k (k>1) 小的元素,那么就取 pivot1 = nums1[k/2-1] 和 pivot2 = nums2[k/2-1] 进行比较。 这里的 “/” 表示整除,nums1 中小于等于 pivot1 的元素有 nums1[0 .. k/2-2] 共计 k/2-1 个。 nums2 中小于等于 pivot2 的元素有 nums2[0 .. k/2-2] 共计 k/2-1 个。 取 pivot = min(pivot1, pivot2),两个数组中小于等于 pivot 的元素共计不会超过 (k/2-1) + (k/2-1) <= k-2 个 这样 pivot 本身最大也只能是第 k-1 小的元素 如果 pivot = pivot1,那么 nums1[0 .. k/2-1] 都不可能是第 k 小的元素。把这些元素全部 “删除”,剩下的作为新的 nums1 数组 如果 pivot = pivot2,那么 nums2[0 .. k/2-1] 都不可能是第 k 小的元素。把这些元素全部 “删除”,剩下的作为新的 nums2 数组 由于我们 “删除” 了一些元素(这些元素都比第 k 小的元素要小),因此需要修改 k 的值,减去删除的数的个数

class Solution {
    public double findMedianSortedArrays(int[] nums1, int[] nums2) {
        int n1 = nums1.length;
        int n2 = nums2.length;
        int totalLength = n1 + n2;
        if(totalLength % 2 == 1) { // 最小
            int midIndex = totalLength / 2;
            // 第mindIndex小的数字
            double median = getKthElement(nums1, nums2, midIndex+1);
            return median;
        } else {
            int midIndex1 = totalLength / 2 - 1, midIndex2 = totalLength / 2;
            double median = (getKthElement(nums1, nums2, midIndex1+1) + getKthElement(nums1, nums2, midIndex2+1)) / 2.0;
            return median;
        }
    }

    public int getKthElement(int[] nums1, int[] nums2, int k) {
        /* 主要思路:要找到第 k (k>1) 小的元素,那么就取 pivot1 = nums1[k/2-1] 和 pivot2 = nums2[k/2-1] 进行比较
         * 这里的 "/" 表示整除
         * nums1 中小于等于 pivot1 的元素有 nums1[0 .. k/2-2] 共计 k/2-1 个
         * nums2 中小于等于 pivot2 的元素有 nums2[0 .. k/2-2] 共计 k/2-1 个
         * 取 pivot = min(pivot1, pivot2),两个数组中小于等于 pivot 的元素共计不会超过 (k/2-1) + (k/2-1) <= k-2 个
         * 这样 pivot 本身最大也只能是第 k-1 小的元素
         * 如果 pivot = pivot1,那么 nums1[0 .. k/2-1] 都不可能是第 k 小的元素。把这些元素全部 "删除",剩下的作为新的 nums1 数组
         * 如果 pivot = pivot2,那么 nums2[0 .. k/2-1] 都不可能是第 k 小的元素。把这些元素全部 "删除",剩下的作为新的 nums2 数组
         * 由于我们 "删除" 了一些元素(这些元素都比第 k 小的元素要小),因此需要修改 k 的值,减去删除的数的个数
         */
        int length1 = nums1.length, length2 = nums2.length;
        int index1 = 0, index2 = 0;
        
        while(true) {
            if(index1==length1)
                return nums2[index2+k-1];
            if(index2==length2)
                return nums1[index1+k-1];
            if(k==1)
                return Math.min(nums1[index1],nums2[index2]);
            
            int half = k /2;
            int newIndex1 = Math.min(index1+half, length1) - 1;
            int newIndex2 = Math.min(index2+half, length2) - 1;
            int pivot1 = nums1[newIndex1], pivot2 = nums2[newIndex2];
            if(pivot1<=pivot2){
                k -= (newIndex1-index1+1);
                index1 = newIndex1+1;
            }else{
                k -= (newIndex2-index2+1);
                index2 = newIndex2+1;
            }
        }
    }

    // public int getKthElement(int[] nums1, int[] nums2, int k) {
    //     int index1 = 0, index2 = 0;
    //     int length1 = nums1.length, length2 = nums2.length;

    //     // 这里的二分需要一直循环
    //     while(true){
    //         // 先看nums1, nums2哪个数组先排除完了
    //         if (index1 == length1)
    //             return nums2[index2+k-1];
    //         if (index2 == length2)
    //             return nums1[index1+k-1];
    //         if(k==1){
    //             return Math.min(nums1[index1], nums2[index2]);
    //         }

    //         int half = k/2;
    //         int half_index1 = Math.min(index1 + half - 1, length1-1);
    //         int half_index2 = Math.min(index2 + half - 1, length2-1);
    //         int pivot1 = nums1[half_index1], pivot2 = nums2[half_index2];
    //         if(pivot1 < pivot2){
    //             k -= (half_index1 - index1+1);
    //             index1 = half_index1+1;
    //         } else {
    //             k -= (half_index2 - index2+1);
    //             index2 = half_index2+1;
    //         }
    //     }
    // }
}

每日练习(不分题型)

765 情侣牵手

题目描述:N 对情侣坐在连续排列的 2N 个座位上,想要牵到对方的手。 计算最少交换座位的次数,以便每对情侣可以并肩坐在一起。 一次交换可选择任意两人,让他们站起来交换座位。

人和座位用 0 到 2N-1 的整数表示,情侣们按顺序编号,第一对是 (0, 1),第二对是 (2, 3),以此类推,最后一对是 (2N-2, 2N-1)。

这些情侣的初始座位  row[i] 是由最初始坐在第 i 个座位上的人决定的。

解题

用贪心的方法解决比较方便,可以制定按顺序让每张沙发上情侣开心的策略。对于每张沙发,找到沙发上第一个人的情侣,如果不在同一个沙发上,就把沙发上的第二人换成第一个人的情侣。 如果一个人的编号为 x,那么他的情侣的编号为 x ^ 1^在这里是异或操作。对于每张沙发上的第一个人 x = row[i],找到他们的同伴所在的位置 row[j],将 row[j] 和 row[i + 1] 互相交换。

class Solution {
public:
    int minSwapsCouples(vector<int>& row) {
        int res=0;
        int len = row.size();
        for(int i=0;i<len; i = i+2){
            int x = row[i];
            if(row[i+1] == (x^1))
                continue;
             
            for(int j=i+2;i<len;j++){
                if(row[j]==(x^1)){
                    row[j]=row[i+1];
                    row[i+1]=x^1;
                    break;
                }
            }
            res++;
        }
        return res;
    }
};

680 验证回文字符串 5月19日

题目描述:给定一个非空字符串s,最多删除一个字符。判断是否能成为回文字符串。

解题思路:两个指针low,high分别从头和从尾向中间靠拢,遇到不一样字符时候,只需要字符串中 [low,high-1]和[low+1,high]这两个子字符串其中一个是回文就行了。

题目链接

class Solution {
public:
    bool validPalindrome(string s) {
        int low = 0, high = s.size()-1;

        while(low<high){
            if(s[low]==s[high]){
                low++;high--;
            }else{
                return check(s,low+1,high) || check(s,low,high-1);
            }
        }
        return true;
    }
private:
    bool check(string s, int low, int high)
    {
        while(low<high){
            if(s[low]!=s[high])
                return false;
            low++;high--;
        }
        return true;
    }
};

1371 每个元音包含偶数次的最长子字符串 5月20日

题目描述:给你一个字符串 s ,请你返回满足以下条件的最长子字符串的长度:每个元音字母,即 ‘a’,’e’,’i’,’o’,’u’ ,在子字符串中都恰好出现了偶数次。

解题思路:将5个元音字母出现次数的奇偶视为一种状态,共有32种状态。

而如果子串 [0,i] 与字串 [0,j] 状态相同,那么字串 [i+1,j] 的状态一定是 00,因此可以记录每个状态第一次出现的位置,此后再出现该状态时相减即可。 需要注意状态 00 首次出现的位置应该设定为 -1。

在计算状态的时候可以利用异或运算。

class Solution {
public:
    int findTheLongestSubstring(string s) {
        vector<int> pre(32,INT_MAX);
        pre[0]=-1;
        const int N=s.size();
        int cur=0;
        int ans=0;
        for(int i=0;i<N;++i){
            switch(s[i]){
                case 'a':cur^=1;break;
                case 'e':cur^=2;break;
                case 'i':cur^=4;break;
                case 'o':cur^=8;break;
                case 'u':cur^=16;break;
                default:break;
            }
            if(pre[cur]==INT_MAX) pre[cur]=i;
            else ans=max(ans,i-pre[cur]);
        }
        return ans;
    }
};

105 从前序与中序遍历序列构造二叉树 5月22日

题目描述:剑指Offer原题,根据一棵树的前序遍历与中序遍历构造二叉树。 题目链接

class Solution {
public:
    TreeNode* buildTree(vector<int>& preorder, vector<int>& inorder) {
        int preorder_len = preorder.size();
        int inorder_len = inorder.size();

        if(inorder_len==0)
            return nullptr;
        
        unordered_map<int,int> kv;

        for(int i=0;i<inorder_len;i++)
        {
            kv[inorder[i]]=i;
        }

        TreeNode* root = build(preorder, 0,preorder_len-1,inorder, 0, inorder_len-1,kv);
        return root;
    }

    TreeNode* build(vector<int>& preorder, int preorder_start, int preorder_end, vector<int>& inorder, int inorder_start, int inorder_end, unordered_map<int,int>& kv)
    {
        if(preorder_start > preorder_end)
            return nullptr;

        int root_val = preorder[preorder_start];
        TreeNode* root = new TreeNode(root_val);
        int inorder_rootindex = kv[root_val];

        if(preorder_start == preorder_end)
            return root;
        
        int left_count = inorder_rootindex - inorder_start, right_count = inorder_end - inorder_rootindex;

        root->left = build(preorder, preorder_start+1,preorder_start+left_count, inorder, inorder_start, inorder_start+left_count-1,kv);
        root->right = build(preorder, preorder_start+left_count+1, preorder_end, inorder, inorder_rootindex+1,inorder_end,kv);
        return root;
    }
};

76 最小覆盖子串 5月23日

题目描述:给你一个字符串 S、一个字符串 T,请在字符串 S 里面找出:包含 T 所有字符的最小子串。

题目链接

1.右边界先移动,找到一个满足题意的包含T中所有字符的子区间,然后左边界尝试向右移动,让满足要求的子区间越来越小。直到左区间在向右移动过程中子区间不再包含所有T中的字符。 2.接着左边界放在满足当前最小区间位置向右一格,右边界继续向右移动,重复1的过程。

class Solution {
    Map<Character, Integer> ori = new HashMap<Character, Integer>();
    Map<Character, Integer> cnt = new HashMap<Character, Integer>();

    public String minWindow(String s, String t) {
        int tLen = t.length();
        for (int i = 0; i < tLen; i++) {
            char c = t.charAt(i);
            ori.put(c, ori.getOrDefault(c, 0) + 1);
        }
        int l = 0, r = -1;
        int len = Integer.MAX_VALUE, ansL = -1, ansR = -1;
        int sLen = s.length();
        while (r < sLen) {//向右移动
            ++r;
            if (r < sLen && ori.containsKey(s.charAt(r))) {
                cnt.put(s.charAt(r), cnt.getOrDefault(s.charAt(r), 0) + 1);
            }
            while (check() && l <= r) {
                if (r - l + 1 < len) {
                    len = r - l + 1;
                    ansL = l;
                    ansR = l + len;
                }
                if (ori.containsKey(s.charAt(l))) {
                    cnt.put(s.charAt(l), cnt.getOrDefault(s.charAt(l), 0) - 1);
                }
                ++l;
            }
        }
        return ansL == -1 ? "" : s.substring(ansL, ansR);
    }

    public boolean check() {
        Iterator iter = ori.entrySet().iterator(); 
        while (iter.hasNext()) { 
            Map.Entry entry = (Map.Entry) iter.next(); 
            Character key = (Character) entry.getKey(); 
            Integer val = (Integer) entry.getValue(); 
            if (cnt.getOrDefault(key, 0) < val) {
                return false;
            }
        } 
        return true;
    }
}

331. 验证二叉树的前序遍历 5月28日

题目描述:”9,3,4,#,#,1,#,#,2,#,6,#,#”验证是否是前序遍历。

解题思路:二叉树中任意一个节点或者空孩子节点都要占据一个槽位。二叉树的建立也伴随着槽位数量的变化。开始时只有一个槽位,如果根节点是空节点,就只消耗掉一个槽位,如果根节点不是空节点,除了消耗一个槽位,还要为孩子节点增加两个新的槽位。之后的节点也是同理。

     _9_
    /   \
   3     2
  / \   / \
 4   1  ##  6
/ \ / \   / \
## ## ## ##   ## #
class Solution {
    public boolean isValidSerialization(String preorder) {
        int slots=1;

        for(String node: preorder.split(",")){
            slots--;

            if(slots<0)
                return false;
            
            if(!node.equals("#")){
                slots = slots + 2;
            }
        }

        return slots==0;
    }
}

394. 字符串解码 5月28日

题目描述:给定一个经过编码的字符串,返回它解码后的字符串。 示例

s = "3[a]2[bc]", 返回 "aaabcbc".
s = "3[a2[c]]", 返回 "accaccacc".
s = "2[abc]3[cd]ef", 返回 "abcabccdcdcdef".
class Solution {
    int ptr;
    public String decodeString(String s) {
        LinkedList<String> stk = new LinkedList<String>();
        ptr = 0;

        while(ptr < s.length()){
            char cur = s.charAt(ptr);

            if(Character.isDigit(cur)){ //是数字
                String digits = getDigits(s);
                stk.addLast(digits);
            }else if (Character.isLetter(cur) || cur == '[') { //是字符和左括号
                // 获取一个字母并进栈
                stk.addLast(String.valueOf(s.charAt(ptr++))); 
            }else{  //右括号,要开始出栈了
                ++ptr;
                LinkedList<String> sub = new LinkedList<String>();
                while(!"[".equals(stk.peekLast())){
                    sub.addLast(stk.removeLast());
                }
                Collections.reverse(sub);
                stk.removeLast();
                int repTime = Integer.parseInt(stk.removeLast());
                StringBuffer t = new StringBuffer();
                String o = getString(sub);
                while(repTime-- > 0){
                    t.append(o);
                }
                stk.addLast(t.toString());
            }
        }

        return getString(stk);
    }

    private String getDigits(String s){
        StringBuffer ret = new StringBuffer();
        while(Character.isDigit(s.charAt(ptr))){
            ret.append(s.charAt(ptr++));
        }
        return ret.toString();
    }

    public String getString(LinkedList<String> v) {
        StringBuffer ret = new StringBuffer();
        for (String s : v) {
            ret.append(s);
        }
        return ret.toString();
    }
}

28. 实现strStr().

题目描述:找出字符串B在字符串A中第一次出现的位置

解题思路:KMP算法,KMP算法目的就是:在出错时,利用原有的匹配信息,尽量减少重新匹配的次数。 (1) 搜索串与主串匹配过程中如果遇到不匹配字符,重复利用之前已经匹配过可免匹配的部分。 (2) KMP 算法(三个人名)思路就是找到搜索串每个字符之前可以免匹配的最大长度。 *** 比如搜索串 abaabb 匹配到最后一个字符不匹配,则前面可免匹配的部分为 ab ,从第三个字符开始继续。

构造next数组。参考视频。next数组的作用是为了重复利用之前已经匹配过,可以免匹配的部分。

class Solution {
    public int strStr(String haystack, String needle) {
        if(needle.isEmpty()){
            return 0;
        }
        char [] h = haystack.toCharArray();
        char [] n = needle.toCharArray();
        int h_len = h.length;
        int n_len = n.length;

        if(n_len > h_len)
            return -1;

        int[] next = getNext(needle);
        int i=0,j=0;

        while(i < h_len && j < n_len){
            if(j==-1 || h[i]==n[j]){
                i++;j++;
            }else{
                j = next[j];
            }
        }

        if(j==n_len)
            return i-j;
        else{
            return -1;
        }
    }

    private int[] getNext(String needle) {
        char[] n = needle.toCharArray();

        int[] next = new int[n.length];
        next[0]=-1;
        int j=0;
        int k=-1;

        while(j < n.length-1){
            if(k==-1 || n[k]==n[j]){
                next[++j]=++k;
            }else{
                k = next[k];
            }
        }
        return next;
    }
}

101. 对称二叉树 5月31日

题目描述:给定一个二叉树,检查它是否是镜像对称的。

解题思路:剑指Offer原题

class Solution {
    public boolean isSymmetric(TreeNode root) {
        if(root==null)
            return true;
        
        return help(root.left, root.right);
    }

    private boolean help(TreeNode root1, TreeNode root2){
        if(root1==null && root2==null)
            return true;
        
        if(root2==null || root1==null){
            return false;
        }

        if(root1.val != root2.val)
            return false;
        
        return help(root1.left, root2.right) && help(root1.right, root2.left);
    }
}

1431. 拥有最多糖果的孩子 6月1日

题目链接

题目说明:给你一个数组 candies 和一个整数 extraCandies ,其中 candies[i] 代表第 i 个孩子拥有的糖果数目。

对每一个孩子,检查是否存在一种方案,将额外的 extraCandies 个糖果分配给孩子们之后,此孩子有 最多 的糖果。注意,允许有多个孩子同时拥有 最多 的糖果数目

class Solution {
    public List<Boolean> kidsWithCandies(int[] candies, int extraCandies) {
        int len = candies.length;
        int maxcandie = 0;
        for(int i=0;i<len;i++){
            maxcandie = Math.max(maxcandie, candies[i]);
        }
        List<Boolean> res = new ArrayList<Boolean>();
        for(int i=0;i<len;i++){
            res.add(candies[i]+extraCandies>=maxcandie);
        }
        return res;
    }
}

剑指Offer64. 求1+2+…+n 6月2日

题目链接

Java递归

class Solution {
    public int sumNums(int n) {
        boolean flag = n > 0 && (n += sumNums(n - 1)) > 0;
        return n;
    }
}
class Solution {
public:
    int sumNums(int n) {
        return quick_sum(n+1,n) >> 1; 
    }

    int quick_sum(int a, int n)//a*n
    {
        int ans=0;
        (n>0) && (ans = ans + quick_sum(a<<1,n>>1) );
        (n & 1) && (ans = ans + a);

        return ans;
    }
};

837. 新21点 6月3日

题目链接 以 0 分开始,并在她的得分少于 K 分时抽取数字。 抽取时,她从 [1, W] 的范围中随机获得一个整数作为分数进行累计,其中 W 是整数。 每次抽取都是独立的,其结果具有相同的概率。

当爱丽丝获得不少于 K 分时,她就停止抽取数字。 爱丽丝的分数不超过 N 的概率是多少?

class Solution {
    public double new21Game(int N, int K, int W) {
        if (K == 0) {
            return 1.0;
        }
        double[] dp = new double[K + W + 1];
        for (int i = K; i <= N && i < K + W; i++) {
            dp[i] = 1.0;
        }
        dp[K - 1] = 1.0 * Math.min(N - K + 1, W) / W;
        for (int i = K - 2; i >= 0; i--) {
            dp[i] = dp[i + 1] - (dp[i + W + 1] - dp[i + 1]) / W;
        }
        return dp[0];
    }
}

128. 最长连续序列 6月6日

题目链接 给定一个未排序的整数数组,找出最长连续序列的长度。

要求算法的时间复杂度为 O(n)。

用并查集做

class Solution {
    public int longestConsecutive(int[] nums) {
        Set<Integer> num_set = new HashSet<Integer>();

        for(int num:nums){
            num_set.add(num);
        }

        int longest = 0;
        for(int num: num_set){
            if(!num_set.contains(num-1)){
                int cur = num;
                int curLong = 1;
                while(num_set.contains(cur+1)){
                    cur++;
                    curLong++;
                }

                longest = Math.max(longest, curLong);
            }
        }

        return longest;
    }
}

990. 等式方程的可满足性 6月8日

题目链接

题目描述:给定一个由表示变量之间关系的字符串方程组成的数组,每个字符串方程 equations[i] 的长度为 4,并采用两种不同的形式之一:”a==b” 或 ”a!=b”。在这里,a 和 b 是小写字母(不一定不同),表示单字母变量名。

只有当可以将整数分配给变量名,以便满足所有给定的方程时才返回 true,否则返回 false。

解题思路

class Solution {
    public boolean equationsPossible(String[] equations) {
        int len = equations.length;
        if(len<1)return true;
        int [] parent = new int[26];//并查集初始化,26个字母
        for(int i=0;i<26;i++){
            parent[i]=i;
        }
        for(String str:equations){
            if(str.charAt(1)=='='){
                int index1 = str.charAt(0) - 'a';
                int index2 = str.charAt(3) - 'a';
                union(parent, index1, index2);
            }
        }

        for(String str: equations){
            
            if(str.charAt(1)=='!'){
                int index1 = str.charAt(0) - 'a';
                int index2 = str.charAt(3) - 'a';
              
                if(find(parent,index1) == find(parent,index2) ){
                    return false;
                }
            }
        }
        return true;
    }

    private void union(int[] parent, int x, int y){
        int a = find(parent,x);
        int b = find(parent,y);
        if(a == b)
            return;
        parent[a] = b;
        //需要考虑rank的并查集,防止算法退化
        /*
        if(rank[a]<rank[b])
            parent[a] = b;
        else{
            parent[b] = a;
            if(rank[a] == rank[b]) rank[a] = rank[a]+1;
        }
        */
    }

    private int find(int[] parent, int x){
        return parent[x]==x ? x : (parent[x] = find(parent,parent[x]));
    }
}

1014. 最佳观光组合 6月17日

题目链接

给定正整数数组 A,A[i] 表示第 i 个观光景点的评分,并且两个景点 i 和 j 之间的距离为 j - i。

一对景点(i < j)组成的观光组合的得分为(A[i] + A[j] + i - j):景点的评分之和减去它们两者之间的距离。

返回一对观光景点能取得的最高分。

class Solution {
    public int maxScoreSightseeingPair(int[] A) {
        int mx = A[0]+0;
        int ans = 0;
        for(int j=1;j<A.length;j++){
            ans = Math.max(ans, mx+A[j]-j);
            mx = Math.max(mx,A[j]+j);
        }
        return ans;
    }
}

221. 最大正方形 6月17日

在一个由 0 和 1 组成的二维矩阵内,找到只包含 1 的最大正方形,并返回其面积。

示例:

输入:

1 0 1 0 0 1 0 1 1 1 1 1 1 1 1 1 0 0 1 0

输出: 4

动态规划,dp[i][j]表示 (i,j)为右下角的最大正方向边长。状态转移规律也比较好找。

class Solution {
    public int maximalSquare(char[][] matrix) {
        int maxSide = 0;
        if (matrix == null || matrix.length == 0 || matrix[0].length == 0) {
            return maxSide;
        }
        int rows = matrix.length, columns = matrix[0].length;
        int[][] dp = new int[rows][columns];

        for(int i=0; i<rows; i++)
            for(int j=0; j<columns; ++j){
                if(matrix[i][j]=='1'){
                    if(i==0 || j==0){
                        dp[i][j]=1;
                        maxSide = Math.max(maxSide,dp[i][j]);
                    }else{
                        dp[i][j]=Math.min(dp[i-1][j],Math.min(dp[i][j-1],dp[i-1][j-1]))+1;
                        maxSide = Math.max(maxSide,dp[i][j]);
                    }
                }
            }

        int ans = maxSide * maxSide;
        return ans;
    }
}

79. 单词搜索 6月18日

题目链接 剑指 offer原题04 DFS回溯搜索

class Solution {
    public boolean exist(char[][] board, String word) {
        int rows = board.length;
        if (rows<=0)
            return false;
        int cols = board[0].length;

        for(int i = 0;i<rows; i++)
            for(int j=0; j<cols; j++){
                if(search(board, word,rows,cols, 0, i, j))
                    return true;
            }

        return false;
    }

    private boolean search(char[][] board, String word,int rows, int cols, int k,int i, int j){
        if(i<0 || i>=rows || j<0 || j>=cols || board[i][j]!=word[k])
            return false;

        char tmp = board[i][j];
        board[i][j]='/';
        int[] dx = {-1,1,0,0};
        int[] dy = {0,0,-1,1};

        for(int k=0;k<4;k++){
            if(search(board,word,rows,cols,k+1,i+dx[k],j+dy[k]))
                return true;
        }
        board[i][j]=tmp;
        return false;
    }
}

1028. 从先序遍历还原二叉树 6月18日

题目链接

有一个题目中,重建二叉树,需要中序遍历和先序遍历才能重建二叉树。这里重建时只有一个先序遍历字符串就可以重建,是因为这个题目当中可以判断哪个节点是叶节点。思想同37题序列化二叉树

/**
 * Definition for a binary tree node.
 * public class TreeNode {
 *     int val;
 *     TreeNode left;
 *     TreeNode right;
 *     TreeNode(int x) { val = x; }
 * }
 */
class Solution {
    String internalString;
    public TreeNode recoverFromPreorder(String S) {
        internalString = S;
        return helper(0);
    }

    public TreeNode helper(int curdepth){
        int i = 0;
        if("".equals(internalString))
            return null;
        while(internalString.charAt(i)=='-')
            i++;
        if(i!=curdepth){
            return null;
        }
        String tmpString = internalString.substring(i);
        int index = tmpString.indexOf('-');
        String node = index==-1? tmpString : tmpString.substring(0,index);
        internalString = index==-1? "" : tmpString.substring(index);
        TreeNode root = new TreeNode(Integer.parseInt(node));
        root.left = helper(curdepth+1);
        root.right = helper(curdepth+1);

        return root;
    }
}

42. 接雨水 6月18日

题目链接

给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。

首先想清楚这个雨水高度是怎么算的,每个位置能放的雨水高度 = 两边最大高度的较小值 - 当前高度。

  • 动态规划方法,用动态规划计算每个位置两边最大高度。
    class Solution {
      public int trap(int[] height) {
          int length = height.length;
          if(length<=0)
              return 0;
    
          int ans=0;
          int[] right_max = new int[length];
          int[] left_max = new int[length];
          left_max[0]=height[0];
          right_max[length-1]=height[length-1];
          for(int i=1;i<length;i++){
              left_max[i] = Math.max(left_max[i-1],height[i]);
          }
          for(int i=length-2;i>=0;i--){
              right_max[i] = Math.max(right_max[i+1],height[i]);
          }
          for(int i=0;i<length;i++){
              ans += (Math.min(left_max[i],right_max[i]) - height[i]);
          }
          return ans;
      }
    }
    
  • 双指针方法,想办法一次完成遍历 如果一端有更高的条形块(例如右端),积水的高度依赖于当前方向的高度(从左到右)。当我们发现另一侧(右侧)的条形块高度不是最高的,我们则开始从相反的方向遍历(从右到左)。 我们必须在遍历时维护 $\text{left_max}$ 和 $\text{right_max}$ ,但是我们现在可以使用两个指针交替进行,实现 1 次遍历即可完成。
class Solution {
    public int trap(int[] height) {
        int left=0,right=height.length-1;

        int left_max = 0, right_max = 0;
        int ans = 0;
        while(left<right){
            if( height[left] < height[right] ){
                if(height[left] >= left_max) 
                    left_max = height[left];
                else
                    ans += (left_max-height[left]);
                left++;
            }else{
                if(height[right] >= right_max) 
                    right_max = height[right];
                else
                    ans+=(right_max - height[right]);
                right--;
            }
        }
        return ans;
    }
}

139. 单词拆分 6月18日

题目链接

给定一个非空字符串 s 和一个包含非空单词列表的字典 wordDict,判定 s 是否可以被空格拆分为一个或多个在字典中出现的单词。

  • 暴力,超时
    class Solution {
      public boolean wordBreak(String s, List<String> wordDict) {
          if("".equals(s))
              return true;
          for(int i = 0; i < s.length(); i++){
              String str = s.substring(0,i+1);
              if(wordDict.contains(str) && wordBreak(s.substring(i+1), wordDict))
                  return true;
          }
    
          return false;
      }
    }
    
  • 回溯+记忆化,一个Boolean数组保存后缀是否可以匹配
    class Solution {
      public boolean wordBreak(String s, List<String> wordDict) {
          return help(s, new HashSet(wordDict), new Boolean[s.length()], 0);
      }
    
      public boolean help(String s, List<String> wordDict, Boolean [] memo, int start){
            
          if(start==s.length())
              return true;
    
          if(memo[start]!=null)//true和false都保存
              return memo[start];
    
          for(int i = start; i < s.length(); i++){
              String str = s.substring(start,i+1);
              if(wordDict.contains(str) && help(s, wordDict, memo, i+1)){
                  return memo[start]=true;
              }
          }
    
          return memo[start] = false;   
      }
    }
    

785. 判断二分图 6月20日

给定一个无向图graph,当这个图为二分图时返回true。

如果我们能将一个图的节点集合分割成两个独立的子集A和B,并使图中的每一条边的两个节点一个来自A集合,一个来自B集合,我们就将这个图称为二分图。

Graph将会以邻接表方式给出,graph[i]表示图中与节点i相连的所有节点。每个节点都是一个在0到graph.length-1之间的整数。这图中没有自环和平行边: graph[i] 中不存在i,并且graph[i]中没有重复的值。

深度优先搜索着色。用栈深度优先搜索。

class Solution {
    public boolean isBipartite(int[][] graph) {
        
        int n = graph.length;

        int[] color = new int[n];
        Arrays.fill(color, -1);

        for(int i=0;i<n;i++){
            if(color[i]==-1){
                Stack<Integer> stack = new Stack<>();
                stack.push(i);
                color[i]=0;
                
                while(!stack.empty()){
                    Integer node = stack.pop();
                    for(int nn : graph[node]){
                        if(color[nn]==-1){
                            stack.push(nn);
                            color[nn]=color[node]^1;
                        }else if(color[nn]==color[node]){
                            return false;
                        }
                    }
                }
            }
        }
        return true;
    }
}

210. 课程表 II 6月20日

现在你总共有 n 门课需要选,记为 0 到 n-1。 在选修某些课程之前需要一些先修课程。 例如,想要学习课程 0 ,你需要先完成课程 1 ,我们用一个匹配来表示他们: [0,1] 给定课程总量以及它们的先决条件,返回你为了学完所有课程所安排的学习顺序。 可能会有多个正确的顺序,你只要返回一种就可以了。如果不可能完成所有课程,返回一个空数组。

使用 DFS 来实现拓扑排序,使用一个栈存储后序遍历结果,这个栈的逆序结果就是拓扑排序结果。

证明:对于任何先序关系:v->w,后序遍历结果可以保证 w 先进入栈中,因此栈的逆序结果中 v 会在 w 之前。

public int[] findOrder(int numCourses, int[][] prerequisites) {
    List<Integer>[] graphic = new List[numCourses];
    for (int i = 0; i < numCourses; i++) {
        graphic[i] = new ArrayList<>();
    }
    for (int[] pre : prerequisites) {
        graphic[pre[0]].add(pre[1]);
    }
    Stack<Integer> postOrder = new Stack<>();
    boolean[] globalMarked = new boolean[numCourses];
    boolean[] localMarked = new boolean[numCourses];
    for (int i = 0; i < numCourses; i++) {
        if (hasCycle(globalMarked, localMarked, graphic, i, postOrder)) {
            return new int[0];
        }
    }
    int[] orders = new int[numCourses];
    for (int i = numCourses - 1; i >= 0; i--) {
        orders[i] = postOrder.pop();
    }
    return orders;
}

private boolean hasCycle(boolean[] globalMarked, boolean[] localMarked, List<Integer>[] graphic,
                         int curNode, Stack<Integer> postOrder) {

    if (localMarked[curNode]) {
        return true;
    }
    if (globalMarked[curNode]) {
        return false;
    }
    globalMarked[curNode] = true;
    localMarked[curNode] = true;
    for (int nextNode : graphic[curNode]) {
        if (hasCycle(globalMarked, localMarked, graphic, nextNode, postOrder)) {
            return true;
        }
    }
    localMarked[curNode] = false;
    postOrder.push(curNode);
    return false;
}
class Solution {
    List<List<Integer>> edges;
    int[] res;
    int[] visited;
    int index;
    public int[] findOrder(int numCourses, int[][] prerequisites) {
        edges = new ArrayList<List<Integer>>();
        for(int i=0; i< numCourses; i++){
            edges.add(new ArrayList<>());
        }
        visited = new int[numCourses];
        index = numCourses-1;
        res = new int[numCourses];
        for(int [] info:prerequisites){
            edges.get(info[1]).add(info[0]);
        }
        for(int i=0;i<numCourses;i++){
            if(visited[i]==0){
                if(!dfs(i)){
                    return new int[0];
                }
            }
        }
        return res;
    }

    private boolean dfs(int u){
        visited[u] = 1;
        
        for(int v: edges.get(u)){
            if(visited[v]==0){
                if(!dfs(v))
                    return false;
            }else if(visited[v]==1){
                return false;
            }
        }
        res[index--] = u;
        visited[u] = 2;
        return true;
    }
}

124. 二叉树中的最大路径和 6月21日

题目链接 每一层返回的是当前节点的贡献值,通过两边子节点的最大贡献值,计算经过当前root节点的最大路径和。维护一个全局变量搜索(有的题目,递归函数返回的值只是一个辅助,有时候答案需要用一个全局的变量去存储)。

class Solution {
    int ans = 1<<31;
    public int maxPathSum(TreeNode root) {
        if(root==null)return 0;
        contribution(root);
        return ans;
    }

    private int contribution(TreeNode root){
        if(root==null){
            return 0;
        }
        int left = Math.max(0, contribution(root.left));
        int right = Math.max(0, contribution(root.right));

        ans = Math.max(ans , root.val+left+right);
        return root.val + Math.max(left,right);
    }
}

面试题 16.18. 模式匹配

题目链接

挺难的

class Solution {
    public boolean patternMatching(String pattern, String value) {
        String s[]=new String[2];
        return solve(s,pattern,0,value,0);
    }
    /**
     * 回溯遍历设置a,b的对应值,尝试每一种可能。
     * @param s   s[0]=a对应的字符串 s[1]=b对应的字符串
     * @param pattern 模式串
     * @param index1 模式串匹配位置
     * @param value 匹配串(待匹配的字符串)
     * @param index2 匹配串匹配位置
     * @return
     */
    public boolean solve(String []s,String pattern,int index1,String value,int index2){
        //匹配完成
        if(index1==pattern.length()&&index2==value.length()) return true;
        //注意匹配串匹配位置等于长度的时候也可以继续匹配,因为模式串的a,b可以匹配空串。
        if(index1>=pattern.length()||index2>value.length()) return false;
        int num=pattern.charAt(index1)-'a';
        if(s[num]==null){
            //从当前尝试a或b对应的字符串的每一种可能
            for(int i=index2;i<=value.length();i++){
                s[num]=value.substring(index2,i);
                //(!s[num].equals(s[num^1]))  [是指a,b对应的字符串不可相等]
                if(!s[num].equals(s[num^1])&&solve(s,pattern,index1+1,value,i)) return true;
            }
            //失配时应将设置过的对应字符串为空
            s[num]=null;
            return false;
        }else{
            //若此前a或b已有对应的字符串匹配了,则查看当前位置时候能够匹配上。
            int end=index2+s[num].length();
            if(end> value.length()||!value.substring(index2,end).equals(s[num])) return false;
            return solve(s,pattern,index1+1,value,end);
        }
    }
}

718. 最长重复子数组 7月1日

题目链接 给两个整数数组 A 和 B ,返回两个数组中公共的、长度最长的子数组的长度。

class Solution {
    public int findLength(int[] A, int[] B) {
        int aLength = A.length;
        int bLength = B.length;
        int[][] dp = new int[aLength+1][bLength+1];

        dp[0][0] = 0;
        int ans = 0;
        for(int i=1; i <= aLength; i++)
            for(int j=1; j <= bLength; j++){
                if(A[i-1]==B[j-1]){
                    dp[i][j]=dp[i-1][j-1]+1;
                }else{
                    dp[i][j]=0;
                }
                ans = Math.max(ans, dp[i][j]);
            }
        
        return ans;
    }
}

378. 有序矩阵中第K小的元素 7月2日

题目链接

解题思路:二分查找

class Solution {
    public int kthSmallest(int[][] matrix, int k) {
        //二分法查找
        int n = matrix.length;
        if(n<=0){
            return -1;
        }
        //假设K永远有效

        int left = matrix[0][0];
        int right = matrix[n-1][n-1];
        int mid;
        while(left<right){
            mid = (left+right)/2;
            if(check(matrix, k, n,mid)){
                right = mid;
            }else{
                left = mid+1;
            }
        }
        return left;
    }

    private boolean check(int[][] matrix, int k, int n, int mid){
        int num=0;
        //左下角开始搜索
        int i=n-1;
        int j=0;
        
        while( i>=0 && j<n ){
            if(matrix[i][j]<=mid){
                j++;
                num+=(i+1);
            }else{
                i--;
            }
        }

        return num>=k;
    }
}

108. 将有序数组转换为二叉搜索树 7月3日

题目链接

二叉搜索树的中序遍历是递增的,每次构建只要每次取中间那个节点,就可以保证得到比较平衡的二叉搜索树

/**
 * Definition for a binary tree node.
 * public class TreeNode {
 *     int val;
 *     TreeNode left;
 *     TreeNode right;
 *     TreeNode(int x) { val = x; }
 * }
 */
class Solution {
    public TreeNode sortedArrayToBST(int[] nums) {
        int len = nums.length;
        //中序遍历是递增遍历
        if(len<1)
            return null;
        
        return buildTree(nums, 0, len-1);

    }

    private TreeNode buildTree(int[] nums, int left, int right){

        if(left <= right){
            int mid = (left+right) / 2;
            TreeNode root = new TreeNode(nums[mid]);
            root.left = buildTree(nums, left, mid-1);
            root.right = buildTree(nums, mid+1, right); 
            return root;
        }

        return null;
    }
}

32. 最长有效括号 7月4日

题目链接 给定一个只包含 ‘(‘ 和 ‘)’ 的字符串,找出最长的包含有效括号的子串的长度。

栈的思想,栈底是「最后一个没有被匹配的右括号的下标」

class Solution {
    public int longestValidParentheses(String s) {
        //最长有效括号
        //栈的思想的话,栈底是「最后一个没有被匹配的右括号的下标」
        Stack<Integer> stc = new Stack<>();
        stc.push(-1);
        int len = s.length();
        int ans = 0;
        for(int i=0;i<len;i++){
            if('('==s.charAt(i)){
                stc.push(i);
            }else{
                stc.pop();
                if(stc.empty()){
                    stc.push(i);
                }else{
                    ans = Math.max(ans, i-stc.peek());
                }
            }
        }
        return ans;
    }
}

动态规划的思想去做


class Solution {
    public int longestValidParentheses(String s) {
        int n = s.length();
        int[] dp = new int[n];
        int ans = 0;
        // 默认初始化为0
        for(int i=1; i < n; i++) {
            if (s.charAt(i) == ')') {
                if (s.charAt(i-1) == '(') {
                    dp[i] = (i-2 >= 0 ? dp[i-2] : 0) + 2;
                } else if (dp[i-1] > 0) {
                    if (i-dp[i-1]-1 >=0 && s.charAt(i - dp[i-1] - 1) == '(') {
                        dp[i] = (i-dp[i-1]-2 >= 0 ? dp[i-dp[i-1]-2] : 0) + dp[i-1] + 2;
                    }
                }
                ans = Math.max(ans, dp[i]);
            }
        }
        return ans;
    }
}

44. 通配符匹配 7月5日

题目链接

题目描述: 给定一个字符串 (s) 和一个字符模式 (p) ,实现一个支持 ’?’ 和 ’*‘ 的通配符匹配。

’?’ 可以匹配任何单个字符。 ‘*’ 可以匹配任意字符串(包括空字符串)。

解题思路: 动态规划

class Solution {
    public boolean isMatch(String s, String p) {
        int m = s.length();
        int n = p.length();
        boolean[][] dp = new boolean[m + 1][n + 1];
        dp[0][0] = true;
        for (int i = 1; i <= n; ++i) {
            if (p.charAt(i - 1) == '*') {
                dp[0][i] = true;
            } else {
                break;
            }
        }
        for (int i = 1; i <= m; ++i) {
            for (int j = 1; j <= n; ++j) {
                if (p.charAt(j - 1) == '*') {
                    dp[i][j] = dp[i][j - 1] || dp[i - 1][j];
                } else if (p.charAt(j - 1) == '?' || s.charAt(i - 1) == p.charAt(j - 1)) {
                    dp[i][j] = dp[i - 1][j - 1];
                }
            }
        }
        return dp[m][n];
    }
}

63. 不同路径 II 7月6日

题目链接

题目描述: 一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为“Start” )。

机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角(在下图中标记为“Finish”)。

现在考虑网格中有障碍物。那么从左上角到右下角将会有多少条不同的路径?

回溯搜索,超时

class Solution {
    private int ans=0;
    public int uniquePathsWithObstacles(int[][] obstacleGrid) {
        if(obstacleGrid.length <=0 ){
            return 0;
        }
        if(obstacleGrid[0].length<=0){
            return 0;
        }
        this.ans = 0;
        help(obstacleGrid, 0, 0);
        return ans;
    }

    private void help(int[][] obstacleGrid, int i, int j)
    {
        if(i<0 || i>= obstacleGrid.length || j<0 || j>=obstacleGrid[0].length || obstacleGrid[i][j]!=0)
            return;
        if(i == obstacleGrid.length-1 && j == obstacleGrid[0].length-1){
            this.ans++;
            return;
        }
        obstacleGrid[i][j]=-1;

        int[] dx = {0,1};
        int[] dy = {1,0};
        for(int k=0;k<2;k++)
        {
            help(obstacleGrid, i+dx[k], j+dy[k]);
        }
        obstacleGrid[i][j]=0;
    }
}

动态规划+状态压缩

class Solution {
    
    public int uniquePathsWithObstacles(int[][] obstacleGrid) {
        int n = obstacleGrid.length, m = obstacleGrid[0].length;
        int[] dp = new int[m];
        
        //注意有障碍
        dp[0] = obstacleGrid[0][0] == 0 ? 1 : 0;

        for(int i=0;i<n;i++)
            for(int j=0;j<m;j++)
            {
                if(obstacleGrid[i][j]==1){
                    dp[j]=0;
                    continue;
                }
                if(j>=1 && obstacleGrid[i][j-1]==0)
                    dp[j]+=dp[j-1];
            }
        return dp[m-1];
    }
}

6112. 路径总和 7月7日

题目链接

题目描述: 给定一个二叉树和一个目标和,判断该树中是否存在根节点到叶子节点的路径,这条路径上所有节点值相加等于目标和。 说明: 叶子节点是指没有子节点的节点。

解题思路:

/**
 * Definition for a binary tree node.
 * public class TreeNode {
 *     int val;
 *     TreeNode left;
 *     TreeNode right;
 *     TreeNode(int x) { val = x; }
 * }
 */
class Solution {
    public boolean hasPathSum(TreeNode root, int sum) {
        
        if(root==null)
            return false;
        
        if(root.left == null && root.right==null)
            return sum == root.val;
        
        return hasPathSum(root.left, sum-root.val) || hasPathSum(root.right, sum-root.val);
        
    }
}

面试题 16.11. 跳水板 7月8日

题目链接

题目描述: 你正在使用一堆木板建造跳水板。有两种类型的木板,其中长度较短的木板长度为shorter,长度较长的木板长度为longer。你必须正好使用k块木板。编写一个方法,生成跳水板所有可能的长度。

返回的长度需要从小到大排列。

示例:

输入: shorter = 1 longer = 2 k = 3 输出: {3,4,5,6}

解题思路:

class Solution {
    public int[] divingBoard(int shorter, int longer, int k) {
        if (k == 0) {
            return new int[0];
        }
        if (shorter == longer) {
            return new int[]{shorter * k};
        }

        int[] ans = new int[k+1];
        for(int i=0;i<=k;i++){
            ans[i] = shorter*(k-i) + longer*i;
        }
        return ans;
    }
}

面试题 17.13. 恢复空格 7月9日

题目链接

题目描述: 哦,不!你不小心把一个长篇文章中的空格、标点都删掉了,并且大写也弄成了小写。像句子”I reset the computer. It still didn’t boot!”已经变成了”iresetthecomputeritstilldidntboot”。在处理标点符号和大小写之前,你得先把它断成词语。当然了,你有一本厚厚的词典dictionary,不过,有些词没在词典里。假设文章用sentence表示,设计一个算法,把文章断开,要求未识别的字符最少,返回未识别的字符数。

注意:本题相对原题稍作改动,只需返回未识别的字符数

输入: dictionary = [“looked”,”just”,”like”,”her”,”brother”] sentence = “jesslookedjustliketimherbrother” 输出: 7 解释: 断句后为”jess looked just like tim her brother”,共7个未识别字符

解题思路: Trie树 + 动态规划

官方题解重的Trie树数据结构

class Trie{
    Trie[] next;
    boolean isEnd;
    public Trie(){
        next = new Trie[26];
        isEnd = false;
    }

    public void insert(String s){
        Trie curPos = this;
        for(int i=0; i<s.length(); i++ ){
            int t = s.charAt(i) - 'a';
            if(curPos.next[t]==null){
                curPos.next[t] = new Trie();
            }
            curPos = curPos.next[t];
        }
        curPos.isEnd = true;
    }
}

动归过程

dp[i]表示前i个字符最少字符未匹配数

class Solution {
    private class Trie{
        Trie [] next;
        boolean isEnd;

        public Trie(){
            next = new Trie[26];
            isEnd = false;
        }

        public void insert(String s){
            int len = s.length();
            Trie cur = this;
            for(int i=len-1;i>=0;i--){
                int t = s.charAt(t) - 'a';
                if(cur.next[t]==null){
                    cur.next[t] = new Trie();
                }
                cur = cur.next[t];
            }
            cur.isEnd = true;
        }
    }
    public int respace(String[] dictionary, String sentence) {
        // 尽量让未匹配的字符数最少
        int n = sentence.length();
        int[] dp = new int[n+1];
        Trie root = new Trie();
        for(String s: dictionary){
            root.insert(s);
        }
        Arrays.fill(dp, Integer.MAX_VALUE);
        dp[0] = 0;
        for(int i=1;i<=n;i++){
            dp[i] = dp[i-1]+1;
            Trie cur = root;
            for(int j=i;j>=1;j--){
                int t = sentence.charAt(j-1) - 'a';
                if(cur.next[t]==null){
                    break;
                }else if(cur.next[t].isEnd){
                    dp[i] = Math.min(dp[i], dp[j-1]);
                }
                // if(dp[i]==0) break; 这里可以更快
                cur = cur.next[t];
            }
        }
        return dp[n];
    }
}

84. 柱状图中最大的矩形 7月10日

题目链接

题目描述: 给定 n 个非负整数,用来表示柱状图中各个柱子的高度。每个柱子彼此相邻,且宽度为 1 。

求在该柱状图中,能够勾勒出来的矩形的最大面积。

解题思路: 使用单调栈,还可以再单调栈的基础上进行常数优化。

class Solution {
    public int largestRectangleArea(int[] heights) {
        Stack<Integer> stack = new Stack<>();
        stack.push(-1);
        int n = heights.length;
        if (n==0) return 0;
        int ans = 0;

        for(int i=0;i<n ;i++){
            while( !(stack.peek()).equals(-1) && heights[stack.peek()] > heights[i]  ){
                int topHeight = heights[stack.pop()];
                int width = i - stack.peek()-1;
                ans = Math.max(ans, topHeight*width);
            }
            stack.push(i);
        }
        while(!(stack.peek()).equals(-1) ){
            int topHeight = heights[stack.pop()];
            int width = n - stack.peek()-1;
            ans = Math.max(ans, topHeight*width);
        }
        return ans;
    }
}

309. 最佳买卖股票时机含冷冻期 7月10日

题目链接

题目描述: 给定一个整数数组,其中第 i 个元素代表了第 i 天的股票价格 。​

设计一个算法计算出最大利润。在满足以下约束条件下,你可以尽可能地完成更多的交易(多次买卖一支股票):

你不能同时参与多笔交易(你必须在再次购买前出售掉之前的股票)。 卖出股票后,你无法在第二天买入股票 (即冷冻期为 1 天)。

解题思路: 状态机 二维数组 加入 冷冻期

class Solution {
    public int maxProfit(int[] prices) {
        int len = prices.length;
        if (len < 2) return 0;

        //0表示本不持有,1表示持有,2表示当天卖出,不持有
        int[][] dp = new int[len][3];  //用二维数组记录当天各种情况的最优解(收益的最大值)
        dp[0][1] = -prices[0];  //第一天若持有,则收益为负;不持有则收益为零

        for (int i = 1; i < len; i++){
            dp[i][0] = Math.max(dp[i - 1][0], dp[i - 1][2]);  //当天的本不持有可以由前一天本不持有或前一天卖出得到
            dp[i][1] = Math.max(dp[i - 1][1], dp[i - 1][0] - prices[i]);  //当天的持有可以由前一天的持有或前一天的本不持有-当天的股票价格得到, 即买进一只股票
            dp[i][2] = dp[i - 1][1] + prices[i];  //当天卖出可以由前一天的持有+当天的股票价格得到, 即卖出手中的股票
        }
        return Math.max(dp[len - 1][0], dp[len - 1][2]);  //返回最后一天不持有股票的状态,此处可以得到收益的最大值
    }
}

315. 计算右侧小于当前元素的个数 7月11日

题目链接

题目描述: 给定一个整数数组 nums,按要求返回一个新数组 counts。数组 counts 有该性质: counts[i] 的值是  nums[i] 右侧小于 nums[i] 的元素的数量。

示例:

解题思路:

用归并排序,因为归并排序会更改元素位置,用一个初始下标数组记录就可以了。

174. 地下城游戏 7月12日

题目链接

题目描述:

解题思路:

从左上到右下的方向不满足动态规划的无后效性,所以才用从右下到左上的遍历方式。

class Solution {
    public int calculateMinimumHP(int[][] dungeon) {
        
        int n = dungeon.length, m = dungeon[0].length;
        int[][] dp = new int[n+1][m+1];
        
        for(int i = 0; i<=n; i++){
            Arrays.fill(dp[i], Integer.MAX_VALUE);
        }
        // 从dp[n][m]倒过来开始dp。
        dp[n-1][m] = 1;
        dp[n][m-1] = 1;
        for(int i=n-1; i >= 0 ;i--)
            for(int j=m-1; j >= 0; j--){
                /*
                if(dungeon[i][j]<=0){
                    dp[i][j] = Math.min(dp[i+1][j], dp[i][j+1]) - dungeon[i][j];
                }else{
                    dp[i][j] = Math.max( Math.min(dp[i+1][j],dp[i][j+1] ) - dungeon[i][j] , 1);
                }*/
                //简化
                int minn = Math.min(dp[i+1][j], dp[i][j+1]) - dungeon[i][j];
                dp[i][j] = Math.max(minn, 1);
            }
        return dp[0][0];
    }
}

174. 地下城游戏 7月12日

题目链接

题目描述: 给定一个三角形,找出自顶向下的最小路径和。每一步只能移动到下一行中相邻的结点上。

相邻的结点 在这里指的是 下标 与 上一层结点下标 相同或者等于 上一层结点下标 + 1 的两个结点。

解题思路:

杨辉三角类型的动态规划。

class Solution {
    public int minimumTotal(List<List<Integer>> triangle) {
        int ans = 0;
        int n = triangle.size();
        int[] dp = new int[n];
        for(int i=0;i<n;i++){
            dp[i] = triangle.get(n-1).get(i);
        }
        for(int i=n-2;i>=0;i--){
            for(int j=0;j<=i;j++){
                dp[j] = Math.min(dp[j], dp[j+1]) + triangle.get(i).get(j);
            }
        }
        return dp[0];
    }
}

96. 不同的二叉搜索树 7月15日

题目链接

给定一个整数 n,求以 1 … n 为节点组成的二叉搜索树有多少种?

输入: 3 输出: 5 解释: 给定 n = 3, 一共有 5 种不同结构的二叉搜索树:

动态规划,G[n] = $\sum^{n}_{i}$ G[i-1] $\times$ G[n-i]

class Solution {
    public int numTrees(int n) {
        int G[] = new int[n+1];
        G[0] = 1;
        G[1] = 1;
        for(int i=2;i<=n;i++){
            for(int j=1;j<=i;j++){
                G[i]+=G[j-1]*G[i-j];
            }
        }
        return G[n];
    }
}

350. 两个数组的交集 II 7月16日

题目链接

给定两个数组,编写一个函数来计算它们的交集。

思路:

1 排序后,分别从两个数组的头部开始匹配。根据两个数组当前数字之间的大小,移动各自数组中的index。

class Solution {
    public int[] intersect(int[] nums1, int[] nums2) {
        Arrays.sort(nums1);
        Arrays.sort(nums2);
        int length1 = nums1.length, length2 = nums2.length;
        int[] intersection = new int[Math.min(length1, length2)];
        int index1 = 0, index2 = 0, index = 0;
        while (index1 < length1 && index2 < length2) {
            if (nums1[index1] < nums2[index2]) {
                index1++;
            } else if (nums1[index1] > nums2[index2]) {
                index2++;
            } else {
                intersection[index] = nums1[index1];
                index1++;
                index2++;
                index++;
            }
        }
        return Arrays.copyOfRange(intersection, 0, index);
    }
}

2 HashMap

class Solution {
    public int[] intersect(int[] nums1, int[] nums2) {
        if (nums1.length > nums2.length) {
            return intersect(nums2, nums1);
        }
        Map<Integer, Integer> map = new HashMap<Integer, Integer>();
        for (int num : nums1) {
            int count = map.getOrDefault(num, 0) + 1;
            map.put(num, count);
        }
        int[] intersection = new int[nums1.length];
        int index = 0;
        for (int num : nums2) {
            int count = map.getOrDefault(num, 0);
            if (count > 0) {
                intersection[index++] = num;
                count--;
                if (count > 0) {
                    map.put(num, count);
                } else {
                    map.remove(num);
                }
            }
        }
        return Arrays.copyOfRange(intersection, 0, index);
    }
}

剑指 Offer 21. 调整数组顺序使奇数位于偶数前面 7月19日

题目链接

输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有奇数位于数组的前半部分,所有偶数位于数组的后半部分。

思路: 用快排的代码作,一次遍历。本质是双指针。

class Solution {
    public int[] exchange(int[] nums) {
        int n = nums.length;
        if (n==0)
            return nums;
        int x = nums[n-1];
        int i = -1;
        for(int j=0;j<n-1;j++){
            if((nums[j] & 1)==1){
                i++;
                swap(nums,i,j);
            }
        }
        swap(nums,i+1,n-1);
        return nums;
    }
    private void swap(int[] nums, int i, int j){
        int tmp = nums[i];
        nums[i]=nums[j];
        nums[j] = tmp;
    }
}

312. 戳气球 7月19日

题目链接

有 n 个气球,编号为0 到 n-1,每个气球上都标有一个数字,这些数字存在数组 nums 中。

现在要求你戳破所有的气球。如果你戳破气球 i ,就可以获得 nums[left] * nums[i] * nums[right] 个硬币。 这里的 left 和 right 代表和 i 相邻的两个气球的序号。注意当你戳破了气球 i 后,气球 left 和气球 right 就变成了相邻的气球。

求所能获得硬币的最大数量。

思路:

自底向上,不是戳爆气球,而是从0添加气球🎈。

class Solution {
    public int maxCoins(int[] nums) {
        int n = nums.length;
        int[][] dp = new int[n + 2][n + 2];
        int[] val = new int[n + 2];
        val[0] = val[n + 1] = 1;
        for (int i = 1; i <= n; i++) {
            val[i] = nums[i - 1];
        }
        for(int i=n-1; i>=0;i-- )
            for(int j=i+2;j<=n+1;j++)
                for(int k=i+1;k<=j-1;k++){
                    int sum = dp[i][k]+dp[k][j];
                    sum += val[i]*val[k]*val[j];
                    dp[i][j] = Math.max(dp[i][j], sum);
                }
        return dp[0][n + 1];
    }
}
class Solution {
public:
    int maxCoins(vector<int>& nums) {
        int n = nums.size();
        vector<vector<int> > dp(n+2, vector<int>(n+2,0));//dp初始化为0
        vector<int> val(n+2);
        val[0]=val[n+1]=1;
        for(int i=1; i<=n; i++){
            val[i]=nums[i-1];
        }

        for(int i=n-1; i>=0;i--)
            for(int j=i+2;j<=n+1; j++)
                for(int k=i+1;k<j;k++){
                    int sum = val[i]*val[k]*val[j];
                    sum += dp[i][k]+dp[k][j];
                    dp[i][j]=max(sum, dp[i][j]);
                }
            
        return dp[0][n+1];
    }
};

329. 矩阵中的最长递增路径 7月26日

题目链接

超时的代码

class Solution {
    private int ans = 0;
    private int[][] vis;
    public int longestIncreasingPath(int[][] matrix) {
        int n = matrix.length;
        if(n<=0)
            return 0;
        int m = matrix[0].length;
        if(m<=0)
            return 0;
        vis = new int[n][m];
        for(int i=0;i<n;i++)
            for(int j=0;j<m;j++)
            {
                dfs(matrix,i,j,0,null);
            }
        
        return ans;
        
    }

    private void dfs(int[][] matrix, int i, int j, int k, Integer last){
        if(i<0 || i>=matrix.length || j<0 || j>=matrix[0].length || vis[i][j]==1 || (last!=null && matrix[i][j]<=last))
        {
            ans = Math.max(ans, k);
            return;
        }
            
        
        
        int []dx = {0,0,1,-1};
        int[] dy = {1,-1,0,0};
        vis[i][j]=1;
        for(int p=0;p<4;p++){
            dfs(matrix, i+dx[p],j+dy[p], k+1, matrix[i][j]);
        }
        vis[i][j]=0;
    }
}

增加记忆性

class Solution {
    public int[][] directions = { {-1,0},{1,0},{0,-1},{0,1}};
    public int n, m;
    public int longestIncreasingPath(int[][] matrix) {
        n = matrix.length;
        if(n<=0)
            return 0;
        m = matrix[0].length;
        if(m<=0)
            return 0;
        int[][] memo = new int[n][m];
        int ans=0;
        for(int i=0;i<n;i++)
            for(int j=0;j<m;j++)
            {
                ans = Math.max(ans,dfs(matrix,i,j,memo));
            }
        
        return ans;
        
    }

    private int dfs(int[][] matrix, int i, int j, int[][] memo){
       if(memo[i][j] != 0){
           return memo[i][j];
       }
       ++memo[i][j];
       for(int [] dir:directions){
           int ii = i + dir[0];
           int jj = j + dir[1];
           if(ii>=0 && ii<n && jj>=0 && jj<m && matrix[ii][jj]>matrix[i][j]){
               memo[i][j] = Math.max(memo[i][j], dfs(matrix,ii,jj,memo)+1);
           }
       }
       return memo[i][j];
    }
}

343. 整数拆分 7月30日

题目链接

给定一个正整数 n,将其拆分为至少两个正整数的和,并使这些整数的乘积最大化。 返回你可以获得的最大乘积。

贪心,剑指Offer切绳子,尽量拆成长度为3的短绳子。

class Solution {
    public int integerBreak(int n) {
        if(n<3){
            return 1;
        }
        if(n==3){
            return 2;
        }
        int ans=1;
        while(n>4){
            ans*=3;

            n-=3;
        }
        ans*=n;
        return ans;
    }
}

114. 二叉树展开为链表 8月2日

题目链接

给定一个二叉树,原地将它展开为一个单链表。

class Solution {
    public void flatten(TreeNode root) {
        if(root==null)
            return;
        
        TreeNode right = root.right;
        TreeNode next = root.left;
        if(root.left==null)
            flatten(right);
        else{
            root.left==null;
            root.right = next;
            while(next.right!=null){
                next = next.right;
            }
            next.right = right;
            flatten(root.right);
        }
    }
}

前序遍历递归,在前序遍历的过程中得到

class Solution {
    public void flatten(TreeNode root) {
        List<TreeNode> list = new ArrayList<TreeNode>();
        preorderTraversal(root, list);
        int size = list.size();
        TreeNode cur = root;
        for(int i=1;i<size;i++){
            TreeNode node = list.get(i);
            cur.left=null;
            cur.right=node;
            cur = node;
        }
        
    }

    public void preorderTraversal(TreeNode root, List<TreeNode> list) {
        if(root==null)
            return;
        list.add(root);
        preorderTraversal(root.left, list);
        preorderTraversal(root.right, list);
    }
}

二叉树非递归遍历

前序遍历非递归

class Solution{
    public void flatten(TreeNode root){
        List<TreeNode> list = new ArrayList<TreeNode>();
        Deque<TreeNode> stack = new LinkedList<TreeNode>();
        TreeNode node = root;
        while(node!=null || !stack.isEmpty()){
            while(node!=null){
                list.add(node);
                stack.push(node);
                node=node.left;
            }
            node=stack.pop();
            node=node.right;
        }
        int size = list.size();
        for(int i=1; i<size;i++){
            TreeNode prev = list.get(i-1);
            prev.left=null;
            prev.right=list.get(i);
        }
    }
}

后序遍历非递归。

  • 思路一: 依次将根结点,右节点,左节点押入栈。根->右->左,和先序遍历的根->右->左,很像,就把先序的左右交换,访问改成压栈,压另一个栈。
class Solution{
    public List<Integer> postorderTraversal(TreeNode root){
        Stack<Integer> stack = new Stack<>();
        List<Integer> list = new ArrayList<>();
        if(root==null) return list;
        TreeNode node = root;
        while(node!=null || !stack.empty()){
            while(node!=null){
                list.add(node);
                stack.push(node);
                node=node.right;
            }
            if(!stack.empty()){
                node=stack.pop();
                node=node.left;
            }
        }
        Collections.reverse(list);
        return list;
    }
}
  • 思路二:

要访问一个节点的条件上一个访问的节点是右儿子。我们可以增加一个变量Prev来判断当前节点Curr的上一个节点与它的关系来执行相应的操作。

  • 若Prev为空(Curr节点是根节点)或者Prev是Curr的父节点,将Curr节点的左孩子和右孩子分别压入栈;
  • 若Prev是Curr的左儿子,则将Curr的右儿子压入栈;
  • 否则Prev是Curr的右儿子,访问Curr;
class Solution{
    public List<Integer> postorderTraversal(TreeNode root){
        Stack<Integer> stack = new Stack<>();
        List<Integer> list = new ArrayList<>();
        if(root==null) return list;
        TreeNode prev = null, cur = null;
        stack.push(root);
        while(!stack.empty()){
            cur = stack.peek();//不要弹出
            if(prev==null || prev.left==cur || prev.right==cur){
                if(cur.left!=null)
                    stack.push(cur.left);
                else if(cur.right!=null)
                    stack.push(cur.right);
            }else if(prev == cur.left){
                if(cur.right!=null)
                    stack.push(cur.right);
            }else{
                list.add(cur);
                stack.pop();
            }
            prev = cur;
        }
        return list;
    }
}

中序遍历非递归

用辅助栈

class Solution{
    public List<Integer> inorderTraversal(TreeNode root){
        List<Integer> list = new ArrayList<>();
        Stack<Integer> stack = new Stack<>();
        if(root!=null) return list;
        TreeNode node = root;
        while(node!=null || !stack.empty()){
            while(node!=null){
                stack.push(node);
                node=node.left;
            }
            if(!stack.empty()){
                node=stack.pop();
                list.add(node); //访问节点
                node=node.right;
            }
        }
        return list;
    }
}

207. 课程表 8月4日

题目链接

你这个学期必须选修 numCourse 门课程,记为 0 到 numCourse-1 。

在选修某些课程之前需要一些先修课程。 例如,想要学习课程 0 ,你需要先完成课程 1 ,我们用一个匹配来表示他们:[0,1]

给定课程总量以及它们的先决条件,请你判断是否可能完成所有课程的学习?

class Solution {
    List<List<Integer>> edges;
    int[] visited;
    boolean valid = true;

    public boolean canFinish(int numCourses, int[][] prerequisites) {

        edges = new ArrayList<List<Integer>>();
        for(int i=0; i<numCourses; ++i){
            edges.add(new ArrayList<Integer>());
        }
        // 用List把 边缘表示。转换成 邻接 图
        visited = new int[numCourses];
        for(int[] info : prerequisites){
            edges.get(info[1]).add(info[0]);
        }

        for(int i=0; i<numCourses && valid; ++i){
            if(visited[i] == 0){
                if(!dfs(i)){
                    return false;
                }
            }
        }
        return valid;
    }

    public boolean dfs(int u){
        visited[u] = 1;
        for(int v : edges.get(u)){
            if(visited[v]==1){
                return false;
            }else if(visited[v]==0){
                if(!dfs(v)){
                    return false;
                }
            }
        }
        visited[u] = 2;
        return true;
    }
}
class Solution {
    List<List<Integer>> edges;
    int[] visited;
    boolean valid = true;

    public boolean canFinish(int numCourses, int[][] prerequisites) {

        edges = new ArrayList<List<Integer>>();
        for(int i=0; i<numCourses; ++i){
            edges.add(new ArrayList<Integer>());
        }
        // 用List把 边缘表示。转换成 邻接 图
        visited = new int[numCourses];
        for(int[] info : prerequisites){
            edges.get(info[1]).add(info[0]);
        }

        for(int i=0; i<numCourses && valid; ++i){
            if(visited[i] == 0){
                if(!dfs(i)){
                    return false;
                }
            }
        }
        return valid;
    }

    public boolean dfs(int u){
        visited[u] = 1;
        for(int v : edges.get(i)){
            if(visited[v]==1){
                return false;
            }else if(visited[v]==0){
                if(!dfs(v)){
                    return false
                }
            }
        }
        visied[u] = 2;
        return true;
    }
}

很多动态规划问题大致可以按照下面5个步骤取解释: 1、状态定义 2、状态转移方程 3、初始化 4、输出 5、思考状态压缩

动态规划原理

都是把大问题拆分成小问题。寻找大问题和小问题的递推关系。解决一个个小问题,最终达到解决原问题的效果。动态规划有记忆性,填表把所有已经解决的子问题答案记录下来,在新问题里需要用到的子问题可以直接提取,避免了重复计算,从而节约了时间,所以在问题满足最优性原理之后,用动态规划解决问题的核心就在于填表,表填写完毕,最优解也就找到。

198 打家劫舍

题目链接

class Solution {
    public int rob(int[] nums) {
        if(nums == null || nums.length == 0)
            return 0;
        int length = nums.length;
        if(length == 1)
            return nums[0];
        int[] dp = new int[length];
        dp[0] = nums[0];
        dp[1] = Math.max(nums[0], nums[1]);
        for(int i=2;i<length;i++){
            dp[i] = Math.max(dp[i-1], dp[i-2]+nums[i]);
        }
        return dp[length-1];
    }
}

213 打家劫舍2

题目链接 题目描述:房屋都围成一圈,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。

解题思路:在打家劫舍1的基础上,说给出的房子是环形排列首尾相接的。区别就是第一栋房子和最后一栋房子不能同时偷。只需要计算两种情况:

  1. 不偷第一个,问题就是在后边n-1个房子可以偷的最大价值。
  2. 不偷最后一个,问题就是在前边n-1个房子可以偷的最大价值。

这样转换就没有首尾连接的约束了,可以直接采用上一题的解法。取两种情况的最大值

class Solution {
public:
    //首尾之能选一个偷
    int rob(vector<int>& nums) {
        //思想很简单,不偷第一家,就在后面n-1家里偷,不偷最后一家,就在前n-1家里偷
        int len=nums.size();
        if(len==0)
            return 0;
        if(len==1)
            return nums[0];

        int ans1 = rob1(nums, 0, len-2);
        int ans2 = rob1(nums, 1, len-1);

        return max(ans1,ans2);
    }

    int rob1(vector<int> &nums, int l, int r){

        int ans = 0;
        int pre=0,prepre=0;

        for(int i=l;i<=r;i++){
            ans = max(prepre+nums[i], pre);
            prepre = pre;
            pre = ans;
        }
        return ans;
    }
};

337 打家劫舍3

题目链接 题目描述:房屋成一个二叉树结构,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。

解题思路:结点r,分为偷和不偷两种情况。

偷r结点:左右儿子不能偷。最大价值等于不偷左儿子的最大价值+不偷右儿子的最大价值+r的价值

不偷r节点: 最大价值等于偷或不偷左儿子的最大价值+偷或不偷右儿子的最大价值。

class Solution {
public:
    typedef pair<int, int> pii;
    int rob(TreeNode* root) {
        // 可行窃地区是二叉树
        if(root==nullptr)
            return 0;
        pii ans = helper(root);
        return max(ans.first, ans.second);

    }
    //偷r结点:左右儿子不能偷。最大价值等于不偷左儿子的最大价值+不偷右儿子的最大价值+r的价值

    //不偷r节点: 最大价值等于**偷或不偷**左儿子的最大价值+**偷或不偷**右儿子的最大价值。
    pii helper(TreeNode* root){
        if(root==nullptr)
            return {0,0};
        //helper 返回 偷 这个结点能取到的最大价值,还有 不偷 这个结点时能取到的最大价值
        pii l = helper(root->left);
        pii r = helper(root->right);

        //偷
        int r0 = l.second+r.second+root->val;
        //不偷当前
        int r1 = max(l.first,l.second)+max(r.first,r.second);
        return make_pair(r0,r1);
    }
};

72 编辑距离

题目链接

题目描述:删除,修改,插入,把word1变成word2的最小编辑距离

思路:动态规划,D[i][j]表示word1的前i个字母和word2的前j个之间的编辑距离。

  • D[i][j-1] 为 A 的前 i 个字符和 B 的前 j - 1 个字符编辑距离的子问题。即对于 B 的第 j 个字符,我们在 A 的末尾添加了一个相同的字符,那么 D[i][j]最小可以为 D[i][j-1] + 1

  • D[i-1][j] 为 A 的前 i - 1 个字符和 B 的前 j 个字符编辑距离的子问题。即对于 A 的第 i 个字符,我们在 B 的末尾添加了一个相同的字符,那么 D[i][j] 最小可以为 D[i-1][j] + 1

  • D[i-1][j-1] 为 A 前 i - 1 个字符和 B 的前 j - 1 个字符编辑距离的子问题。即对于 B 的第 j 个字符,我们修改 A 的第 i 个字符使它们相同,那么 D[i][j] 最小可以为 D[i-1][j-1] + 1。特别地,如果 A 的第 i 个字符和 B 的第 j 个字符原本就相同,那么我们实际上不需要进行修改操作。在这种情况下,D[i][j] 最小可以为 D[i-1][j-1]

如果word1[i]==word2[j]: \(\begin{aligned} D[i][j] &=\min (D[i][j-1]+1, D[i-1][j]+1, D[i-1][j-1]) \\ &=1+\min (D[i][j-1], D[i-1][j], D[i-1][j-1]-1) \end{aligned}\)

如果word1[i]!=word2[j]: \(D[i][j]=1+\min (D[i][j-1], D[i-1][j], D[i-1][j-1])\)

<img  src='/edit-distance.png'>
    int minDistance(string word1, string word2) {
        int n = word1.length();
        int m = word2.length();

        // 有一个字符串为空串
        if (n * m == 0) return n + m;

        // DP 数组
        int D[n + 1][m + 1];

        // 边界状态初始化
        for (int i = 0; i < n + 1; i++) {
            D[i][0] = i;
        }
        for (int j = 0; j < m + 1; j++) {
            D[0][j] = j;
        }

        // 计算所有 DP 值
        for (int i = 1; i < n + 1; i++) {
            for (int j = 1; j < m + 1; j++) {
                int left = D[i - 1][j] + 1;
                int down = D[i][j - 1] + 1;
                int left_down = D[i - 1][j - 1];
                if (word1[i - 1] != word2[j - 1]) left_down += 1;
                D[i][j] = min(left, min(down, left_down));

            }
        }
        return D[n][m];
    }
class Solution {
    public int minDistance(String word1, String word2) {
        int n1 = word1.length();
        int n2 = word2.length();
        int[][] dp = new int[n1+1][n2+1];
        for(int j=1;j<=n2;j++) dp[0][j] = dp[0][j-1]+1;
        for(int i=1;i<=n1;i++) dp[i][0] = dp[i-1][0]+1;

        for(int i=1;i<=n1;i++)
            for(int j=1;j<=n2;j++){
                if(word1.charAt(i-1)==word2.charAt(j-1)) dp[i][j] = dp[i-1][j-1];
                else
                    dp[i][j] = Math.min(Math.min(dp[i-1][j],dp[i][j-1]), dp[i-1][j-1])+1;
            }
        
        return dp[n1][n2];
    }
}

99. 恢复二叉搜索树

题目链接

二叉搜索树中的两个节点被错误地交换。 请在不改变其结构的情况下,恢复这棵树。

Morris遍历,时间复杂度O(N),空间复杂度O(H)

class Solution {
    public void recoverTree(TreeNode root) {
        // 二叉搜索树
        TreeNode x = null, y = null, pred = null, predecessor = null;

        while(root!=null){
            if(root.left!=null){
                predecessor = root.left;
                while(predecessor.right!=null && predecessor.right != root){
                    predecessor = predecessor.right;
                }
                if(predecessor.right == null){
                    predecessor.right = root;
                    root = root.left;
                }else{
                    //左子树遍历完成
                    if(pred!=null && root.val < pred.val){
                        y = root;
                        if(x==null){
                            x = pred;
                        }
                        
                    }
                    pred = root;
                        
                    predecessor.right = null;
                    root = root.right;
                }
            }else{
                if(pred!=null && pred.val > root.val){
                    y = root;
                    if(x==null){
                        x = pred;
                    }
                }
                pred = root;
                root = root.right;
            }
        }
        swap(y,x);
    }

    private void swap(TreeNode node1, TreeNode node2){
        int tmp = node1.val;
        node1.val = node2.val;
        node2.val = tmp;
    }
}

93. 复原IP地址

题目链接

给定一个只包含数字的字符串,复原它并返回所有可能的 IP 地址格式。

有效的 IP 地址正好由四个整数(每个整数位于 0 到 255 之间组成),整数之间用 ‘.’ 分隔。

深度搜索,剪枝,回溯,注意前导0时候的剪枝。

class Solution {
    
    private static final int SEG_COUNT = 4;
    int[] segments;
    List<String> ans = new ArrayList<>();

    public List<String> restoreIpAddresses(String s) {
        segments = new int[SEG_COUNT];
        dfs(s, 0, 0);
        return ans;
    }

    public void dfs(String s, int segId, int segStart) {
        // 如果找到了 4 段 IP 地址并且遍历完了字符串,那么就是一种答案
        if(segId == SEG_COUNT && segStart == s.length()){
            StringBuilder sb = new StringBuilder();
            for( int i=0; i<SEG_COUNT-1; i++ ){
                sb.append(segments[i]);
                sb.append('.');
            }
            sb.append(segments[SEG_COUNT-1]);
            ans.add(sb.toString());
        }

        if( segId == SEG_COUNT || segStart == s.length() )
            return;
        
        if(s.charAt(segStart)=='0'){
            segments[segId] = 0;
            dfs(s,segId+1,segStart+1);
        }

        int cur = 0;
        
        for(int segEnd = segStart; segEnd < segStart+3 && segEnd < s.length() ; segEnd++){
            cur = cur*10 + (s.charAt(segEnd)-'0');
            if( cur>0 && cur <= 0XFF ){ //如果cur==0,也可以直接break
                segments[segId] = cur;
                dfs(s, segId+1, segEnd+1);
            }else{
                break;
            }
        }

    }
}

696. 计数二进制子串

题目链接

题目描述: 给定一个字符串 s,计算具有相同数量0和1的非空(连续)子字符串的数量,并且这些子字符串中的所有0和所有1都是组合在一起的。 重复出现的子串要计算它们出现的次数。

解题思路:

例如0011000110的字符串转换为这样的整数数组;{2,2,3,2,1}

class Solution {
    public int countBinarySubstrings(String s) {
        char[] chars = s.toCharArray();
        int n = chars.length;
        int ans = 0;
        int one = 0;
        int[] count = new int[2];

        int flag = 0;
        int i=0;
        while(i<n){
            while(i<n && flag == chars[i]-'0'){
                count[flag]++;
                i++;
            }
            ans += Math.min(count[1-flag], count[flag]);
            count[1-flag] = 0;
            flag = 1-flag;
        }
        return ans;
    }
}

130. 被围绕的区域 8月11日

题目链接

题目描述: 给定一个二维的矩阵,包含 ‘X’ 和 ‘O’(字母 O)。 找到所有被 ‘X’ 围绕的区域,并将这些区域里所有的 ‘O’ 用 ‘X’ 填充。

解题思路: 和边界的’O’直接或间接连通的’O’就是不用打叉的O。

class Solution {
    private int[][] d = { {0,-1},{0,1},{1,0},{-1,0}};
    public void solve(char[][] board) {
        int n = board.length;
        if(n==0)
            return;
        int m = board[0].length;
        if(m==0)
            return;
        
        for(int i=0;i<n;i++){
            dfs(board, i,0,n,m);
            dfs(board, i, m-1,n,m);
        }
        for(int i=0;i<m ;i++){
            dfs(board, 0,i,n,m);
            dfs(board, n-1,i,n,m);
        }
        for(int i=0;i<n;i++)
            for(int j=0; j<m;j++){
                if(board[i][j]=='A')
                    board[i][j]='O';
                else if(board[i][j]=='O')
                    board[i][j] = 'X';
            }
    }

    private void dfs(char[][] board, int i, int j,int n, int m ){
        if(i<0 || i>=n || j<0 || j>=m || board[i][j]!='O'){
            return ;
        }

        board[i][j] = 'A';
        for(int i=0;i<4;i++){
            dfs(board,i+d[i][0],j+d[i][1],n,m);
        }
    }
}

107. 二叉树的层次遍历 II 9月6日

题目链接

给定一个二叉树,返回其节点值自底向上的层次遍历。 (即按从叶子节点所在层到根节点所在的层,逐层从左向右遍历)

class Solution {
    public List<List<Integer>> levelOrderBottom(TreeNode root) {
        List<List<Integer>> levelOrder = new LinkedList<List<Integer>>();
        if (root == null) {
            return levelOrder;
        }
        Queue<TreeNode> queue = new LinkedList<TreeNode>(); //队列
        queue.offer(root);
        while (!queue.isEmpty()) {
            int size = queue.size();
            List<Integer> level = new ArrayList<Integer>();
            for(int i=0;i<size;i++){
                TreeNode node = queue.poll();
                level.add(node.val);
                if(node.left!=null)
                    queue.offer(node.left);
                if(node.right!=null)
                    queue.offer(node.right);
            }
            levelOrder.add(0, level);
        }
        return levelOrder;
    }
}

77. 组合 9月6日

题目链接

给定两个整数 n 和 k,返回 1 … n 中所有可能的 k 个数的组合。

  • 方法一 找出所有组合,不重复,就按照某种顺序来展开搜索,这样就能做到不重复不遗漏。
class Solution {
    public List<List<Integer>> combine(int n, int k) {
        Stack<Integer> stack = new Stack<>();
        List<List<Integer>> res = new ArrayList<List<Integer>>();
        dfs(res, stack, k, 1,n);
        return res;
    }

    private void dfs(List<List<Integer>> res, Stack<Integer> stack, int k, int start, int n){
        if(n - start + 1 < k){ //减少搜索情况
            return;
        }
        if(k<=0){
            res.add(new ArrayList<>(stack));
            return;
        }
        for(int i = start; i <= n; i++){
            stack.push(i);
            dfs(res, stack, k-1, i+1, n);
            stack.pop(i);
        }
        return;
    }
}
  • 方法二 按每一个数,选还是不选画出二叉树。

class Solution {
    public List<List<Integer>> combine(int n, int k) {
        
        List<List<Integer>> res = new ArrayList<List<Integer>>();
        if(k<=0 || n<k)
            return ans;
        Stack<Integer> stack = new Stack<Integer>();
        dfs(res, stack, k, 1, n);
        return res;
    }

    private void dfs(List<List<Integer>> res, Stack<Integer> stack, int k, int start, int n){
        if(k == 0 ){
            res.add(new ArrayList<Integer>(stack));
            return;
        }

        if( start > n - k + 1 ){
            return;
        }

        // 当前数字不要
        dfs(res, stack, k, start+1, n);
        
        // 要当前数字
        stack.push(start);
        dfs(res, stack, k-1, start+1, n);
        stack.pop();

        return;
    }
}

15. 三数之和 9月14日

题目链接

给你一个包含 n 个整数的数组 nums,判断 nums 中是否存在三个元素 a,b,c ,使得 a + b + c = 0 ?请你找出所有满足条件且不重复的三元组。

注意:答案中不可以包含重复的三元组。 参考

class Solution {
    List<List<Integer>> res;
    public List<List<Integer>> threeSum(int[] nums) {
        res = new ArrayList<>();
        Arrays.sort(nums);
        
        for (int i = 0; i < nums.length; i++) {
            if (nums[i] > 0) break; // min > 0 后续没必要找了
            if (i == 0 || nums[i] != nums[i - 1]) twoSum(nums, i); // 1层去重
        }
        
        return res;
    }
    
    public void twoSum(int[] nums, int index) {
        int i = index + 1, j = nums.length - 1, item = nums[index], target = -item;
        long sum;
     
        while (i < j) {
            sum = (long) nums[i] + (long) nums[j];
            if (sum == target) {
                res.add(Arrays.asList(item, nums[i], nums[j]));
                // 2层去重,111333这种情况,只需要算一次,i,j 移动到 最内部 1,3
                while (i + 1 < j && nums[i+1] == nums[i]) i++;
                while (j - 1 > i && nums[j-1] == nums[j]) j--;
            }
                
            if (sum > target) j--;
            else i++;
        }
    }
}
class Solution {  
    public List<List<Integer>> threeSum(int[] nums) {
        int n = nums.length;
        List<List<Integer>> list = new ArrayList<>();
        Arrays.sort(nums);
        for(int i=0;i<n;i++){
            if(i>0 && nums[i]==nums[i-1]) continue;
            if(nums[i]>0) break;

            int L=i+1;
            int R=n-1;
            while(L<R){
                int sum = nums[i]+nums[L]+nums[R];
                if(sum==0){
                    list.add(Arrays.asList(nums[i],nums[L],nums[R]));
                    //去除结果
                    while(L<R && nums[L]==nums[L+1])L++;
                    while(L<R && nums[R-1]==nums[R])R--;
                    L++;
                    R--;
                }else if(sum>0){
                    R--;
                }else if(sum<0){
                    L++;
                }
            }
        }
        return list;
    }
}

685. 冗余连接 II 9月17日

题目链接

并查集,入度都应该是1,根的入度为0。若有多个答案,返回最后出现的那条边。

  • 不能有入度为2的点
  • 在没有入度为2的点的时候,不能有环
class Solution {
    public int[] findRedundantDirectedConnection(int[][] edges) {
        
        //1 初始化
        int n = edges.length;
        int[] inDegree = new int[n+1];
        for(int[] edge:edges){
            inDegree[edge[1]]++;
        }

        //2 检查入度为2的点
        for(int i=n-1; i>=0; i--){
            if(inDegree[edges[i][1]]>1){
                if(!judgeCircle(edges,n,i)){
                    return edges[i];
                }
            }
        }
        //3 再试试删除入度为1的边
        for(int i=n-1; i>=0; i--){
            if(inDegree[edges[i][1]]==1){
                if(!judgeCircle(edges,n,i)){
                    return edges[i];
                }
            }
        }
        return new int[0];
    }

    //用并查集看看有没有环
    public boolean judgeCircle(int[][] edges, int len, int removeEdgeIndex){
        //
        UnionFind ufset = new UnionFind(len);

        for(int i=0;i<len;i++){
            if(i==removeEdgeIndex) continue;

            if(!ufset.same(edges[i][0], edges[i][1])){
                ufset.union(edges[i][0], edges[i][1]);
            }else{
                return true;//有环
            }
        }
        return false;
    }

    private class UnionFind{
        private int[] parent;

        public UnionFind(int n){
            parent = new int[n+1];

            for(int i=0; i<n+1; i++){
                parent[i] = i;
            }
        }

        public void union(int x, int y){
            x=find(x);
            y=find(y);

            //有向图,x设为y的父节点
            if(x!=y){
                parent[y]=x;
            }
        }

        public boolean same(int x,int y){
            int rootx = find(x);
            int rooty = find(y);

            return rootx==rooty;
        }

        public int find(int x){
            return (parent[x]==x) ? x : (x = find(parent[x]));
        }
    }
}

47. 全排列 II 9月19日

题目链接

给定一个可包含重复数字的序列,返回所有不重复的全排列。

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

class Solution {
    private boolean[] vis;

    public List<List<Integer>> permuteUnique(int[] nums) {
        List<List<Integer> > res = new ArrayList<List<Integer>>();
        List<Integer> aPerm = new ArrayList<>();
        int length = nums.length;
        vis = new boolean[length];
        Arrays.fill(vis, false);
        // 排序是为了更好的去重。
        Arrays.sort(nums);
        backtrace(nums, res, 0, length, aPerm);
        return res;
    }

    private void backtrace(int[] nums, List<List<Integer>> res, int index, int length, List<Integer> aPerm){
        if(index == length){
            res.add(new ArrayList<Integer>(aPerm));
            return;
        }
        for( int i = 0; i < length; i++){
            if(vis[i] || (i>0 && nums[i]==nums[i-1] && !vis[i-1])) continue;//去重复
            vis[i]=true;
            aPerm.add(nums[i]);
            backtrace(nums, res, index+1, length, aPerm);
            vis[i]=false;
            aPerm.remove(index);
        }
    }
}

46. 全排列 9月19日

题目链接

给定一个 没有重复 数字的序列,返回其所有可能的全排列。

class Solution {
    private boolean[] vis;
    public List<List<Integer>> permute(int[] nums) {
        int length = nums.length;
        vis= new boolean[length];
        Arrays.fill(vis, false);
        List<List<Integer>> res = new ArrayList<>();
        List<Integer> aPerm = new ArrayList<>();
        //可以不排序
        backtrace(nums, res, aPerm, 0 , length);
        return res;
    }

    private void backtrace(int[] nums, List<List<Integer>> res, List<Integer> aPerm,int index, int length){
        if(index==length){
            res.add(new ArrayList<>(aPerm));
            return;
        }

        for(int i=0;i<length; i++){
            if(vis[i]) continue;
            vis[i]=true;
            aPerm.add(nums[i]);
            backtrace(nums, res, aPerm, index+1, length);
            vis[i]=false;
            aPerm.remove(index);
        }
    }
}

404. 左叶子之和 9月19日

题目链接

非递归遍历

/**
 * Definition for a binary tree node.
 * public class TreeNode {
 *     int val;
 *     TreeNode left;
 *     TreeNode right;
 *     TreeNode(int x) { val = x; }
 * }
 */
class Solution {
    public int sumOfLeftLeaves(TreeNode root) {
        int ans=0;
        Stack<TreeNode> stack = new Stack<>();
        if(root==null) return ans;
        TreeNode node = root;
        
        while(node!=null || !stack.empty()){
            while(node!=null){
                stack.push(node);
                node=node.left;
            }
            node = stack.pop();
            if(!stack.empty() && node.left==null  && node.right==null && stack.peek().left==node){
                ans+=node.val;
            }
            node=node.right;
        }
        return ans;
    }
}

递归便遍历,深度优先 或 广度优先

class Solution {
    public int sumOfLeftLeaves(TreeNode root) {
        if(root==null) return 0;
        return dfs(root);
    }

    public int dfs(TreeNode root){
        int ans = 0;
        if(root.left!=null){
            ans += isLeaf(root.left) ? root.left.val : dfs(root.left);
        }
        if(root.right!=null){
            ans += isLeaf(root.right) ? 0 : dfs(root.right);
        }

        return ans;
    }

    private boolean isLeaf(TreeNode node){
        return node.left==null && node.right==null;
    }
}

117. 填充每个节点的下一个右侧节点指针 II 9月28日

题目链接

常数级额外空间

/*
// Definition for a Node.
class Node {
    public int val;
    public Node left;
    public Node right;
    public Node next;

    public Node() {}
    
    public Node(int _val) {
        val = _val;
    }

    public Node(int _val, Node _left, Node _right, Node _next) {
        val = _val;
        left = _left;
        right = _right;
        next = _next;
    }
};
*/

class Solution {
    private Node last = null, nextStart = null;
    public Node connect(Node root) {
        if(root==null)
            return null;
        
        Node start = root;
        while(start!=null){
            nextStart=null;
            last=null;
            for(Node p = start; p!=null; p=p.next){
                if(p.left!=null){
                    help(p.left);
                }
                if(p.right!=null){
                    help(p.right);
                }
            }
            start=nextStart;
        }
        return root;
    }

    private void help(Node node){
        if(last!=null){
            last.next=node;
        }
        if(nextStart==null)
            nextStart=node;
        last=node;
    }
}

145. 二叉树的后序遍历 9月29日

笔记:二叉树的后序遍历

非递归方法 TreeNode prev = null, cur = null;

  • prev是空或者prev是cur的父节点,尝试把cur的做左子节点或右子节点压栈
  • prev是cur的左子节点,把cur的右子节点压栈
  • else: prev是cur的右子节点或者其他情况。访问cur。
/**
 * Definition for a binary tree node.
 * public class TreeNode {
 *     int val;
 *     TreeNode left;
 *     TreeNode right;
 *     TreeNode() {}
 *     TreeNode(int val) { this.val = val; }
 *     TreeNode(int val, TreeNode left, TreeNode right) {
 *         this.val = val;
 *         this.left = left;
 *         this.right = right;
 *     }
 * }
 */
class Solution {
    public List<Integer> postorderTraversal(TreeNode root) {
        Stack<TreeNode> stack = new Stack<>();
        List<Integer> list = new ArrayList<>();
        if(root==null) return list;

        TreeNode cur = null, prev = null;
        stack.push(root);
        while(!stack.empty()){
            cur = stack.peek();
            if(prev = null || prev.left==cur || prev.right==cur){
                if(cur.left!=null){
                    stack.push(cur.left);
                }else if(cur.right!=null){
                    stack.push(cur.right);
                }
            }else if(prev==cur.left){
                if(cur.right!=null)
                    stack.push(cur.right);
            }else{
                list.add(cur);
                stack.pop();
            }
            prev = cur;
        }
        return list;
    }
}

Morris遍历

利用树的大量空闲指针,实现空间开销的极限缩减。

class Solution {
    public List<Integer> postorderTraversal(TreeNode root) {
        List<Integer> res = new ArrayList<Integer>();
        if (root == null) {
            return res;
        }

        TreeNode p1 = root, p2 = null;

        while (p1 != null) {
            p2 = p1.left;
            if (p2 != null) {
                while (p2.right != null && p2.right != p1) {
                    p2 = p2.right;
                }
                if (p2.right == null) {
                    p2.right = p1;
                    p1 = p1.left;
                    continue;
                } else {
                    p2.right = null;
                    addPath(res, p1.left);
                }
            }
            p1 = p1.right;
        }
        addPath(res, root);
        return res;
    }

    public void addPath(List<Integer> res, TreeNode node) {
        List<Integer> tmp = new ArrayList<Integer>();
        while (node != null) {
            tmp.add(node.val);
            node = node.right;
        }
        for (int i = tmp.size() - 1; i >= 0; --i) {
            res.add(tmp.get(i));
        }
    }
}

LCP 19. 秋叶收藏集 10月1日

题目链接

求字符串最少调整次数。 状态机思想

class Solution {
    public int minimumOperations(String leaves) {
        int n = leaves.length();
        char[] chars = leaves.toCharArray();
        int[][] dp = new int[n][3];

        dp[0][0] = isYellow(chars[0]);
        dp[1][2] = dp[0][1] = dp[0][2] = Integer.MAX_VALUE;
        
        for(int i=1; i<n; i++){
            dp[i][0] = dp[i-1][0] + isYellow(chars[i]);
            dp[i][1] = Math.min(dp[i-1][0], dp[i-1][1]) + isRed(chars[i]);
            if(i>1)
                dp[i][2] = Math.min(dp[i-1][1], dp[i-1][2]) + isYellow(chars[i]);
        }
        return dp[n-1][2];
    }

    private int isYellow(char c){
        return c=='y' ? 1 : 0;
    }

    private int isRed(char c){
        return c=='r' ? 1 : 0;
    }
}

状态压缩

class Solution {
    public int minimumOperations(String leaves) {
        int n = leaves.length();
        char[] chars = leaves.toCharArray();
        int f1=0, f2=0, f3=0;
        f1 = isYellow(chars[0]);
        f2=f3=Integer.MAX_VALUE;
        
        for(int i=1; i<n; i++){
            if(i>1)
                f3 = Math.min(f2, f3) + isYellow(chars[i]);
            f2 = Math.min(f1, f2) + isRed(chars[i]);
            f1= f1 + isYellow(chars[i]);
        }
        return f3;
    }

    private int isYellow(char c){
        return c=='y' ? 1 : 0;
    }

    private int isRed(char c){
        return c=='r' ? 1 : 0;
    }
}

125. 乘积最大子数组 10月16日

题目链接

因为乘法有正负数的情况,子问题可能不是从i-1这个状态变过来的。 也没有用复杂的动态规划,就是在维护一个数组记录以nums[i]结尾的子数组乘积的最小值。

class Solution {
    public int maxProduct(int[] nums) {
        int n = nums.length;
        int[] Fmax = new int[n];
        int[] Fmin = new int[n];
        System.arraycopy(nums, 0, Fmax, 0, n);
        System.arraycopy(nums, 0, Fmin, 0, n);

        for(int i=1;i<n;i++){
            int num = nums[i];
            Fmax[i] = Math.max(Fmax[i-1]*num, Math.max(Fmin[i-1]*num, num));
            Fmin[i] = Math.min(Fmax[i-1]*num, Math.min(Fmin[i-1]*num, num));
        }
        int ans = Fmax[0];
        for(int i=1;i<n;i++){
            ans = Math.max(ans, Fmax[i]);
        }
        return ans;
    }
}

977. 有序数组的平方 10月16日

题目链接

给定一个按非递减顺序排序的整数数组 A,返回每个数字的平方组成的新数组,要求也按非递减顺序排序。

  1. 简单的计算后排序
  2. 双指针

原数组非递减,把负数和正数分开,双指针类似归并排序

class Solution {
    public int[] sortedSquares(int[] A) {
        int length = A.length;
        int negSize=0;
        for(int i=0;i<length; i++){
            if(A[i]<0) negSize++;
            A[i] = A[i]*A[i];
        }

        int Ptr1 = negSize-1, Ptr2=negSize;
        int[] ans = new int[length];
        
        int index=0;
        while(Ptr1>=0 && Ptr2<length){
            if(A[Ptr1]<=A[Ptr2]){
                ans[index++]=A[Ptr1--];
            }else{
                ans[index++]=A[Ptr2++];
            }
        }

        while(Ptr1>=0 && index<length){
            ans[index++]=A[Ptr1--];
        }
        while(Ptr2<length && index<length){
            ans[index++]=A[Ptr2++];
        }
        return ans;
    }
}

双指针二 使用两个指针分别指向位置 $0$ 和 $n-1$,每次比较两个指针对应的数,选择较大的那个逆序放入答案并移动指针。这种方法无需处理某一指针移动至边界的情况。

class Solution {
    public int[] sortedSquares(int[] A) {
        int n = A.length;
        int[] ans = new int[n];
        for (int i = 0, j = n - 1, pos = n - 1; i <= j;) {
            if (A[i] * A[i] > A[j] * A[j]) {
                ans[pos] = A[i] * A[i];
                ++i;
            } else {
                ans[pos] = A[j] * A[j];
                --j;
            }
            --pos;
        }
        return ans;
    }
}

234. 回文链表 10月23日

题目链接

请判断一个链表是否为回文链表。 快慢指针,从中间找

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode(int x) { val = x; }
 * }
 */
class Solution {
    public boolean isPalindrome(ListNode head) {
        if(head==null)
            return true;
        
        Stack<ListNode> stack = new Stack<>();
        ListNode node1 = head;
        ListNode node2 = head;
        while(node2!=null && node2.next!=null){
            stack.push(node1);
            node1=node1.next;
            node2=node2.next.next;
            if(node2!=null && node2.next==null){
                stack.push(node1);
            }
        }
        while(!stack.empty() && node1!=null){
            ListNode left = stack.pop();
            if(left.val!=node1.val)
                return false;
            node1=node1.next;
        }
        return true;
    }
}

347. 前 K 个高频元素 11月22日

题目链接

堆的方法

class Solution {
    public int[] topKFrequent(int[] nums, int k){
        Map<Integer, Integer> occurences = new HashMap<Integer, Integer>();
        for(int num: nums){
            occurences.put(num, occurences.getOrDefault(num, 0)+1);
        }

        // 最小堆. int[],int[0]是数字,int[1]是出现的次数。 堆里的元素 应该包含 num和频率,因为根据频率比大小
        PriorityQueue<int[]> queue = new PriorityQueue<int[]>(new Comparator<int[]>(){
            public int compare(int[] m, int[] n){
                return m[1] - n[1];
            }
        });
        for(Map.Entry<Integer, Integer> entry: occurences.entrySet()){
            int num = entry.getKey(), count = entry.getValue();
            if(queue.size() >= k){
                if(queue.peek()[1] < count){
                    queue.poll();
                    queue.offer(new int[]{num, count});
                }
            }else{
                queue.offer(new int[]{num, count});
            }
        }
        int[] ret = new int[k];
        for(int i=0; i<k; i++){
            ret[i] = queue.poll()[0];
        }
        return ret;
    }
}

快排

class Solution {
    public int[] topKFrequent(int[] nums, int k) {
        Map<Integer, Integer> occurrences = new HashMap<Integer, Integer>();
        for (int num : nums) {
            occurrences.put(num, occurrences.getOrDefault(num, 0) + 1);
        }

        List<int[]> values = new ArrayList<int[]>();
        for (Map.Entry<Integer, Integer> entry : occurrences.entrySet()) {
            int num = entry.getKey(), count = entry.getValue();
            values.add(new int[]{num, count});
        }
        int[] ret = new int[k];
        qsort(values, 0, values.size() - 1, ret, 0, k);
        return ret;
    }

    public void qsort(List<int[]> values, int start, int end, int[] ret, int retIndex, int k) {
        int picked = (int) (Math.random() * (end - start + 1)) + start;
        Collections.swap(values, picked, start);
        
        int pivot = values.get(start)[1];
        int index = start;
        for (int i = start + 1; i <= end; i++) {
            if (values.get(i)[1] >= pivot) {
                Collections.swap(values, index + 1, i);
                index++;
            }
        }
        Collections.swap(values, start, index);

        if (k <= index - start) {
            qsort(values, start, index - 1, ret, retIndex, k);
        } else {
            for (int i = start; i <= index; i++) {
                ret[retIndex++] = values.get(i)[0];
            }
            if (k > index - start + 1) {
                qsort(values, index + 1, end, ret, retIndex, k - (index - start + 1));
            }
        }
    }
}

437. 路径总和 III 11月29日

Link

class Solution {
    int ans = 0;
    public int pathSum(TreeNode root, int target) {
        
        if(root==null )
            return 0;
        HashMap<Integer, Integer> map = new HashMap<>();
        map.put(0,1);
        pathSum(root, 0, target, map);
        
        return ans;
    }

    private void pathSum(TreeNode root, int cur, int target, HashMap<Integer,Integer> map){
        if(root==null)
            return;
        
        cur = cur + root.val; //现在的和
        ans += map.getOrDefault(cur-target, 0);
        map.put(cur, map.getOrDefault(cur, 0) + 1);
        pathSum(root.left, cur, target, map);
        pathSum(root.right, cur, target, map);
        // 要去掉
        map.put(cur, map.getOrDefault(cur,0)-1);
    }
}

338. 比特位计数 12月7日

Link

打表方法。看起来像动态规划 $res[i] = res[i&(i-1)] + 1$

class Solution {
    public int[] countBits(int num) {
        int[] res = new int[num+1];
        res[0] = 0;
        for(int i=1; i<= num; i++){
            res[i] = res[i&(i-1)] + 1;
        }
        return res;
    }
}

494. 目标和 12月7日 ⭐️

Link

给定一个非负整数数组,a1, a2, …, an, 和一个目标数,S。现在你有两个符号 + 和 -。对于数组中的任意一个整数,你都可以从+-中选择一个符号添加在前面。 返回可以使最终数组和为目标数 S 的所有添加符号的方法数。 提示:

数组非空,且长度不会超过 20 。 初始的数组的和不会超过 1000 。 保证返回的最终结果能被 32 位整数存下。

解析: 类似于 0 1 背包的动态规划。

public class Solution{
    public int findTargetSumWays(int[] nums, int S){
        //初始化都是0
        int[][] dp = new int[nums.length][2001];
        
        dp[0][nums[0]+1000] = 1;
        dp[0][1000-nums[0]] += 1;
        for(int i=1;i<nums.length;i++){
            for(int j=-1000;j<=1000;j++){
                if(dp[i-1][j+1000] > 0){ // 可不可以达到
                    dp[i][j+1000+nums[i]] += dp[i-1][j+1000];
                    dp[i][j+1000-nums[i]] += dp[i-1][j+1000];
                }
            }
        }
        return S > 1000 ? 0 : dp[nums.length - 1][S + 1000];
    }
}

538. 把二叉搜索树转换为累加树 12月7日

给出二叉 搜索 树的根节点,该树的节点值各不相同,请你将其转换为累加树(Greater Sum Tree),使每个节点 node 的新值等于原树中大于或等于 node.val 的值之和。

解析:中序遍历先访问右子节点。

class Solution {
    int sum = 0;
    public TreeNode convertBST(TreeNode root) {
        if(root==null){
            return null;
        }

        convertBST(root.right);
        root.val += sum;
        sum = root.val;
        convertBST(root.left);
        return root;
    }
}

560. 和为K的子数组 12月7日

Link

给定一个整数数组和一个整数 k,你需要找到该数组中和为 k 的连续的子数组的个数。

输入:nums = [1,1,1], k = 2 输出: 2 , [1,1] 与 [1,1] 为两种不同的情况。

前缀和,哈希

class Solution {
    public int subarraySum(int[] nums, int k) {
        HashMap<Integer, Integer> map = new HashMap<>();
        map.put(0, 1);
        int sum = 0;
        int ans = 0;
        for(int num: nums){
            sum += num;
            ans += map.getOrDefault(sum-k,0);
            map.put(sum, map.getOrDefault(sum,0)+1);
        }
        return ans;
    }
}

617. 合并二叉树 12月7日

Link

给定两个二叉树,想象当你将它们中的一个覆盖到另一个上时,两个二叉树的一些节点便会重叠。 你需要将他们合并为一个新的二叉树。合并的规则是如果两个节点重叠,那么将他们的值相加作为节点合并后的新值,否则不为 NULL 的节点将直接作为新二叉树的节点。

/**
 * Definition for a binary tree node.
 * public class TreeNode {
 *     int val;
 *     TreeNode left;
 *     TreeNode right;
 *     TreeNode(int x) { val = x; }
 * }
 */
class Solution {
    public TreeNode mergeTrees(TreeNode t1, TreeNode t2) {

        if(t1 == null && t2 == null){
            return null;
        }
        if(t1 == null)
            return t2;
        if(t2 == null)
            return t1;
        
        t1.val += t2.val;
        t1.right = mergeTrees(t1.right, t2.right);
        t1.left = mergeTrees(t1.left, t2.left);
        return t1;
    }
}

621. 任务调度器 12月9日

Link

给你一个用字符数组 tasks 表示的 CPU 需要执行的任务列表。其中每个字母表示一种不同种类的任务。任务可以以任意顺序执行,并且每个任务都可以在 1 个单位时间内执行完。在任何一个单位时间,CPU 可以完成一个任务,或者处于待命状态。

然而,两个 相同种类 的任务之间必须有长度为整数 n 的冷却时间,因此至少有连续 n 个单位时间内 CPU 在执行不同的任务,或者在待命状态。

你需要计算完成所有任务所需要的 最短时间

解析:贪心队列数组

class Solution{
    
    public int leastInterval(char[] tasks, int n){
        Map<Character, Integer> freq = new HashMap<>();
        
        for(char ch: tasks){
            freq.put(ch, freq.getOrDefault(ch, 0) + 1);
        }
        
        int m = freq.size();
        List<Integer> nextValids = new ArrayList<>();
        List<Integer> res = new ArrayList<>();
        int n  = tasks.length;
        int time = 0;
        for(Map.Entry<Character, Integer> entry : freq.entrySet()){
            int value = entry.getValue();
            nextValids.add(1);
            res.add(value);
        }
        // 遍历n次就结束
        for( int i=0 ; i < n; i++ ){
            time++;
            int minValid = Integer.MAX_VALUE;
            for(int j = 0; j < m; j++){
                if( res.get(j) > 0 && nextValids.get(j) < minValid ){
                    minValid = nextValids.get(j);
                }
            }
            time = Math.max(time, minVlid);
            int best = -1;
            for (int j = 0; j < m; ++j) {
                if (res.get(j) != 0 && nextValids.get(j) <= time) {
                    if (best == -1 || res.get(j) > res.get(best)) {
                        best = j;
                    }
                }
            }
            nextValids.set(best, time + n + 1);
            res.set(best, res.get(best) - 1);
        }
        return time;

    }
}

739. 每日温度 12月10日

Link 哈希表。 例如,给定一个列表 temperatures = [73, 74, 75, 71, 69, 72, 76, 73],你的输出应该是 [1, 1, 4, 2, 1, 1, 0, 0]。

class Solution {
    public int[] dailyTemperatures(int[] T) {
        Stack<Integer> stack = new Stack<Integer>();
        int length = T.length;
        int[] ans = new int[length];
        for(int i=0; i < length; i++){
            int num = T[i];
            while(!stack.empty() && num > T[stack.peek()]){
                int index = stack.pop();
                ans[index] = i-index;
            }
            stack.push(i);
        }
        return ans;
    }
}

406. 根据身高重建队列 12月10日

Link

贪心 解析: hi 升序排列,ki降序排列。 按照排序后的顺序一次进入队列。进入队列的位置找第ki+1个空位置。

class Solution {
    public int[][] reconstructQueue(int[][] people) {
        Arrays.sort(people, new Comparator<int[]>(){
            public int compare(int[] p1, int[] p2){
                if(p1[0] != p2[0]){
                    return p1[0] - p2[0];
                }else{
                    return p2[1] - p1[1];
                }
            }
        });
        int n = people.length;
        int[][] ans = new int[n][];// 为啥这么写
        for(int[] person : people ){
            int space = person[1]+1;//找第p[i]+1个空位置
            for(int i=0;i<n;i++){
                if(ans[i]==null){
                    space--;
                    if(space==0){
                        ans[i] = person;
                        break;
                    }
                }
            }
        }
        return ans;
    }
}

438. 找到字符串中所有字母异位词 12月10日 ⭐️

Link

哈希表

自己写的比较复杂

class Solution {
    public List<Integer> findAnagrams(String s, String p) {
        List<Integer> res = new ArrayList<Integer>();
        int length_p = p.length();
        int length_s = s.length();
        if(length_s < length_p){
            return res;
        }
        // 哈希表
        Map<Character, Integer> map = new HashMap<>();
        for(int i=0; i<length_p; i++){
            map.put(p.charAt(i), map.getOrDefault(p.charAt(i),0)+1);
        }
        int count = map.size();
        
        for(int i=0; i<length_p; i++){
            Integer value = map.get(s.charAt(i));
            if(value!=null){
                map.put(s.charAt(i), value-1);
                if(value==1){
                    count--;
                }else if(value==0){
                    count++;
                }
            }
        }
        if(count==0){
            res.add(0);
        }
        for(int i=length_p; i<length_s; i++)
        {
            int lastRemoveIndex = i - length_p;
            char lastChar = s.charAt(lastRemoveIndex);
            Integer value;
            if((value=map.get(lastChar))!=null){
                // value
                
                map.put(lastChar, value+1);
                if(value+1==0){
                    count--;
                }else if(value==0){
                    count++;
                }
            }
            char curChar = s.charAt(i);
            if((value=map.get(curChar))!=null){
                map.put(curChar, value-1);
                if(value-1==0){
                    count--;
                }else if(value==0){
                    count++;
                }
            }
            if(count==0){
                res.add(lastRemoveIndex+1);
            }
        }
        return res;
    }
}

1203. 项目管理 1月12日

Link

拓扑排序,确定一个序列,满足边\<u->v\>之间的依赖,拓扑不能有环。 题目中有2中拓扑序

  1. 项目和项目之间的依赖
  2. 组和组之间的依赖

首先解决组间的依赖顺序。之后在每个组里面,把属于同一组的任务顺序确定。这两个步骤里面发现有环就表示不能确定顺序。

/*
输入:n = 8, m = 2, group = [-1,-1,1,0,0,1,0,-1], beforeItems = [[],[6],[5],[6],[3,6],[],[],[]]
输出:[6,3,4,1,5,2,0,7]
*/
class Solution {
    public int[] sortItems(int n, int m, int[] group, List<List<Integer>> beforeItems) {
        List<List<Integer>> groupItem = new ArrayList<List<Integer>>();
        for (int i = 0; i < n + m; ++i) {
            groupItem.add(new ArrayList<Integer>());
        }

        List<List<Integer>> groupGraph = new ArrayList<List<Integer>>();
        for(int i=0; i<n+m; i++)
        {
            groupGraph.add(new ArrayList<Integer>());
        }
        List<List<Integer>> itemGraph = new ArrayList<List<Integer>>();
        for(int i = 0; i < n; i++)
        {
            itemGraph.add(new ArrayList<Integer>());
        }

        // 入度数组,组间和组内
        int[] groupDegree = new int[n+m];
        int[] itemDegree = new int[n];

        // id?
        List<Integer> id = new ArrayList<>();
        for(int i=0; i < n + m; i++)
        {
            id.add(i);
        }

        int leftId = m;
        // 无人接手的项目,单独一个group
        for(int i=0; i<n; i++)
        {
            if(group[i]==-1)
            {
                group[i]=leftId++;
            }
            groupItem.get(group[i]).add(i);
        }
        // 建立依赖图关系, 遍历Tasks
        for(int i=0; i < n; i++)
        {
            int curGroupId = group[i];
            for(int item: beforeItems.get(i))
            {
                int beforeGroupId = group[item];
                if(beforeGroupId!=curGroupId)
                {
                    groupGraph.get(beforeGroupId).add(curGroupId);
                    groupDegree[curGroupId]++;
                }else{
                    itemDegree[i]++;
                    itemGraph.get(item).add(i);
                }
            }
        }

        //建立好图过后,组间关系拓扑排序
        List<Integer> groupTopSort = topSort(groupDegree, groupGraph, id);
        if (groupTopSort.size() == 0) {
            return new int[0];
        }
        int[] ans = new int[n];
        int index = 0;
        // 组内关系排序
        for (int curGroupId : groupTopSort) {
            int size = groupItem.get(curGroupId).size();
            if (size == 0) {
                continue;
            }
            List<Integer> res = topSort(itemDegree, itemGraph, groupItem.get(curGroupId));
            if (res.size() == 0) {
                return new int[0];
            }
            for (int item : res) {
                ans[index++] = item;
            }
        }
        return ans;
    }

    public List<Integer> topSort(int[] deg, List<List<Integer>> graph, List<Integer> items) {
        Queue<Integer> queue = new LinkedList<Integer>();
        for (int item : items) {
            if (deg[item] == 0) {
                queue.offer(item);
            }
        }
        List<Integer> res = new ArrayList<Integer>();
        while (!queue.isEmpty()) {
            int u = queue.poll(); 
            res.add(u);
            for (int v : graph.get(u)) {
                if (--deg[v] == 0) {
                    queue.offer(v);
                }
            }
        }
        return res.size() == items.size() ? res : new ArrayList<Integer>();
    }
}

25. K 个一组翻转链表 2月12日

Link

给你一个链表,每 k 个节点一组进行翻转,请你返回翻转后的链表。 k 是一个正整数,它的值小于或等于链表的长度。 如果节点总数不是 k 的整数倍,那么请将最后剩余的节点保持原有顺序。

设置一个函数,反转其中一段链表,输入为这段链表的首尾节点,在里面in-place的反转

/**
 * Definition for singly-linked list.
 * public class ListNode {
 *     int val;
 *     ListNode next;
 *     ListNode() {}
 *     ListNode(int val) { this.val = val; }
 *     ListNode(int val, ListNode next) { this.val = val; this.next = next; }
 * }
 */
class Solution {
    public ListNode reverseKGroup(ListNode head, int k) {
        // 只能使用常数额外空间
        if(k<=1) return head;
        ListNode hair = new ListNode(0);
        hair.next = head;
        ListNode prev = hair;

        while(head!=null){
            ListNode tail = head;
            for(int i=0; i<k-1; i++) {
                tail = tail.next;
                if(tail==null){
                    return hair.next;
                }
            }
            ListNode next = tail.next;
            reverse(head, tail);
            prev.next = tail;
            head.next = next;
            
            prev = head;
            head = next;
        }
        return hair.next;
    }

    // myreverse, 只把从 head 到 tail 这段颠倒顺序,返回新的 head和 tail 节点
    public void reverse(ListNode head, ListNode tail) {
        ListNode prev = tail.next;
        ListNode p = head;

        while(prev != tail) {
            ListNode next = p.next;
            p.next = prev;
            prev = p;
            p = next;
        }        
    }
}

91. 解码方法 4月21日

题目链接

动态规划,自己的代码

class Solution {
    public int numDecodings(String s) {
        // 多少种解码方式. 1~26的数字
        // 动态规划, 暴力搜索肯定可以,为什么这里可以动态规划呢
        int n = s.length();
        if (n==0) return 0;
        int[] dp = new int[n+1];
        dp[0]=1;
        char[] chars = s.toCharArray();

        for(int i=1; i<=n; i++)
        {
            if( chars[i-1]=='0'){
                if(i>=2 && (chars[i-2]=='1'||chars[i-2]=='2')){
                    dp[i] = dp[i-2];
                }
            }else{
                dp[i]=dp[i-1];
                if(i>=2 && chars[i-2]!='0'){
                    int num = string2int(s.substring(i-2,i));
                    if( num>=1 && num<=26){
                        dp[i]+=dp[i-2];
                    }
                }

            }
        }
        return dp[n];
    }

    private int string2int(String s){
        int n=s.length();
        int ans = 0;
        for(int i=0;i<n;i++){
            ans = ans*10+(s.charAt(i)-'0');
        }
        return ans;
    }
}

官方代码

class Solution {
    public int numDecodings(String s) {
        int n = s.length();
        int[] f = new int[n + 1];
        f[0] = 1;
        for (int i = 1; i <= n; ++i) {
            if (s.charAt(i - 1) != '0') {
                f[i] += f[i - 1];
            }
            if (i > 1 && s.charAt(i - 2) != '0' && ((s.charAt(i - 2) - '0') * 10 + (s.charAt(i - 1) - '0') <= 26)) {
                f[i] += f[i - 2];
            }
        }
        return f[n];
    }
}

403. 青蛙过河 4月29日

题目链接

class Solution {
    public boolean canCross(int[] stones) {
        HashMap<Integer, Set<Integer> > map = new HashMap<>();
        for(int i=0; i < stones.length; i++) {
            map.put(stones[i], new HashSet<Integer>());
        }
        map.get(stones[0]).add(0);
        for(int i=0; i < stones.length; i++){
            for(int laststep : map.get(stones[i])){
                for(int step = laststep-1; step<=laststep+1; step++){
                    if(step>0 && map.containsKey(stones[i]+step)){
                        map.get(stones[i]+step).add(step);
                    }
                }
            }
        }
        return map.get(stones[stones.length-1]).size()>0;
    }
}

224. 基本计算器 5月15日

题目链接 本题只有连续的加减法,计算最后的答案可以用类似栈的结构,扫描的过程作同时做括号展开。 本意就是将所有的数字展开相加。 比如,- (- 10 + (6 - (-2 + 3))) 被展开成 (–10) + (-6) + (—2) + (–3)

class Solution {
    public int calculate(String s) {
        char[] chars = s.toCharArray();
        int n = s.length();
        Stack<Integer> stack = new Stack<Integer>();
        int sign = 1;
        stack.push(sign);
        int res = 0;
        for(int i=0; i < n; ) {
            char c = chars[i++];
            switch (c){
                case ' ':
                case '+':break;
                case '-':
                    sign = -sign;
                    break;
                case '(':
                    stack.push(sign);
                    break;
                case ')':
                    stack.pop();
                    sign = stack.peek();
                    break;
                default:
                    int num = c - '0';
                    while(i<n && chars[i]>='0' && chars[i]<='9') {
                        num = num*10 + (chars[i] - '0');
                        i++;
                    }
                    res += sign*num;
                    sign = stack.peek();
                    break;
            }
        }
        return res;
    }
}

772. 基本计算器III 5月15日

题目链接

四则运算,支持括号。逆波兰表达式解法

参考:772 计算器III

class Solution {
    // 包含递归
    public int calculate(String s) {
        int n = s.length();
        int curRes = 0, res = 0;
        long num = 0; // num 非负整数,表示当前正在读取的数字
        char op = '+';

        for( int i=0; i<n; i++) {
            char c = s.charAt(i);
            if (isDigit(c)) {
                num = num * 10 + c - '0';
            }
            else if (c == '(') {
                int j = i, cnt = 0; // 找与当前左括号对应的 ')'
                for(; i < n; i++) {
                    if (s.charAt(i) == '(') ++cnt;
                    if (s.charAt(i) == ')') --cnt;
                    if (cnt == 0) break;
                }
                num = calculate(s.substring( j+1, i ));
            }
            if (c == '+' || c == '-' || c == '*' || c =='/' || i == n - 1) {
                switch (op){
                    case '+': curRes += num; break;
                    case '-': curRes -= num; break;
                    case '*': curRes *= num; break;
                    case '/': curRes /= num; break;
                }
                if (c == '+' || c == '-' || i == n-1) {
                    res += curRes;
                    curRes = 0;
                }
                op = c;
                num = 0;
            }
        }
        return res;
    }

    private boolean isDigit(char c) {
        return c >= '0' && c <= '9';
    }
}

150. 逆波兰表达式求值 5月15日

题目链接

class Solution {
    private Set<String> operators;
    private Stack<Long> operands;
    public int evalRPN(String[] tokens) {
        
        operators = new HashSet<String>(){ {
            add("*");
            add("/");
            add("+");
            add("-");
        }};
        // stack 解法
        operands = new Stack<>();
        
        for( String token : tokens) {
            if(operators.contains(token)){
                HandleOperator(token);
            }else{
                HandleNumber(token);
            }
        }

        return operands.peek().intValue();
    }

    private void HandleOperator(String token){
        Long a = operands.pop();
        Long b = operands.pop();
        switch (token){
            case "+":
                operands.push(a+b);
                break;
            case "-":
                operands.push(b-a);
                break;
            case "*":
                operands.push(b*a);
                break;
            case "/":
                operands.push(b/a);
                break;
        }
    }

    private void HandleNumber(String token){
        Long ret = (long)0;
        int sign = 1;
        int i=0;
        if(token.charAt(i)=='-'){
            sign = -sign;
            i++;
        }
        for(; i < token.length(); i++){
            char c = token.charAt(i);
            ret = ret*10 + c - '0';
        }

        ret = sign*ret;
        operands.push(ret);
    }
}

930. 和相同的二元子数组 7月8日

题目链接

前缀表的方法。扫描数组的过程记录和为x的数量。

class Solution {
    public int numSubarraysWithSum(int[] nums, int goal) {
        int n = nums.length;

        Map<Integer, Integer> prefixSumMap = new HashMap<>();
        // 前缀和
        int sum = 0;
        int res = 0;
        for (int num: nums) {
            prefixSumMap.put(sum, prefixSumMap.getOrDefault(sum, 0)+1);
            sum += num;
            res += prefixSumMap.getOrDefault(sum-goal,0);
        }
        return res;
    }
}

滑动窗口

class Solution {
    public int numSubarraysWithSum(int[] nums, int goal) {
        int n = nums.length;
        int left1 = 0, left2 = 0, right = 0;
        int sum1 = 0, sum2 = 0;
        int ret = 0;
        while (right < n) {
            sum1 += nums[right];
            while (left1 <= right && sum1 > goal) {
                sum1 -= nums[left1];
                left1++;
            }
            sum2 += nums[right];
            while (left2 <= right && sum2 >= goal) {
                sum2 -= nums[left2];
                left2++;
            }
            ret += left2 - left1;
            right++;
        }
        return ret;
    }
}

468. 验证IP地址 7月31日

题目链接

分治验证,也可用正则表达式库,Pattern: ([0-9]|[1-9][0-9]|1[0-9][0-9]|2[0-4][0-9]|25[0-5]) 注意 java 的 split函数:如果limit = -1 。那么返回的数组会保留 空字符串”“。

class Solution {
    public String validIPAddress(String IP) {
        // 编写一个函数来验证输入的字符串是否是有效的 IPv4 或 IPv6 地址。
        if (IP.chars().filter(ch -> ch == '.').count() == 3) {
            return validateIPv4(IP);
        } else if (IP.chars().filter(ch -> ch == ':').count() == 7) {
            return validateIPv6(IP);
        }

        return "Neither";
    }

    private String validateIPv4(String IP) {
        String[] nums = IP.split("\\.",-1); // 正则表达式,.需要转义!! limit = -1 。那么nums数组会保留 空字符串""。
        
        for(String num : nums) {
            if (num.length() == 0 || num.length() >= 4) return "Neither";
            if (num.charAt(0) == '0' && num.length() != 1) return "Neither";
            int number = 0;
            for(char c : num.toCharArray()) {
                if (c < '0' || c > '9') return "Neither";
                number = number * 10 + (c-'0');
                if (number > 255) {
                    return "Neither";
                }
            }
        }
        return "IPv4";
    }

    public String validateIPv6(String IP) {
        String[] nums = IP.split(":", -1);
        
        String hexdigits = "0123456789abcdefABCDEF";
        for (String x : nums) {
            if (x.length() == 0 || x.length() > 4) return "Neither";
            
            for (Character ch : x.toCharArray()) {
                if (hexdigits.indexOf(ch) == -1) return "Neither";
            }
        }
        return "IPv6";
    }
}

任意进制转换

编译后的程序用在我的dotfiles中 使用:

➜ radix FFF 16 32
3VV

➜ radix FFF 16 2
111111111111

➜ radix FFF 16 3
12121200
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstdlib>
using namespace std;

string getQuotientAndRemainder(
    string originNumber,
    int radix_from,
    int radix_to,
    int & remainder) {
    
    remainder = 0;
    int a = 0;
    for(int i=0; i<originNumber.size(); i++) {
        // 任何进制的数,都可以和不同的radix做除法,
        // 模拟 除法手算的方法的,
        // remainder 和 自己的进制radix_from相乘
        a = (remainder * radix_from) + (originNumber[i] - '0');
        originNumber[i] = a / radix_to + '0';
        remainder = a % radix_to;
    }

    int startIndex = 0;
    while (originNumber[startIndex] == '0') {
        startIndex++;
    }
    return originNumber.substr(startIndex);
}

string convert(string originNumber, int radix_from, int radix_to) {

    string result = "";
    char c;
    int remainder = 0;

    while(originNumber.size() != 0) {
        originNumber = getQuotientAndRemainder(originNumber, radix_from, radix_to, remainder);
        result = char(remainder+'0') + result;
    }

    return result;
}

int main(int argc, char*argv[]) {
    // 表示数字的字符串,radix from,radix to
    if (argc < 4) {
        cout<<"radixConverter number radix_from radix_to"<<endl;
        return 0;
    }

    string originalNumber = string(argv[1]);

    int radix_from = atoi(argv[2]);
    int radix_to = atoi(argv[3]);;
    string bistring = convert(originalNumber, radix_from, radix_to);
    cout<<bistring;
}

847. 访问所有节点的最短路径 8月7日

题目链接

class Solution {
    // https://leetcode-cn.com/problems/shortest-path-visiting-all-nodes/

   public int shortestPathLength(int[][] graph) {
        // 用三元组来搜索
        int n = graph.length;
        // 队列
        Queue<int[]> que = new LinkedList<int[]>();
        // 同一个节点u, 和 到达它时节点经过情况,只能被搜索一次,防止重复搜索
        boolean[][] seen = new boolean[n][1 << n]; 

        for(int i=0; i<n; i++) {
            // 三元组, 节点编号,节点经过请款过,当前走过距离
            que.offer(new int[]{i, 1<<i, 0});
            seen[i][1<<i] = true;
        }

        while (!que.isEmpty()) {
            // 如果队列不是空的
            int[] state = que.poll();
            
            int u = state[0];
            int bitmap = state[1];
            int dist = state[2];
            
            if (bitmap == (1<<n) - 1) {
                return dist;
            }
            for (int v : graph[u]) {
                int nBitMap = bitmap | (1<<v);
                if (!seen[v][nBitMap]) {
                    int[] nextState = new int[]{v, nBitMap, dist+1};
                    que.offer(nextState);
                    seen[v][nBitMap] = true;// 因为是广度优先搜索,所以不可能之后遇到这个状态,走过的路径会更小
                }
            }
        }
        return -1;
    }
}

1786. 从第一个节点出发到最后一个节点的受限路径数 8月13日

题目链接

受限路径 为满足 distanceToLastNode(zi) > distanceToLastNode(zi+1) 的一条路径 记忆化搜索,Dijkstra

#include <cstdio>
#include <utility>
#include <algorithm>
#include <queue>
#include <iostream>
#include <vector>
using namespace std;
using ll = long long;

struct DijkNode{
    // 需要知道 该节点当前距离
    int d;
    // 需要知道该节点是谁
    int u;
    
    bool operator > (const DijkNode& a) const
    {
        return d > a.d;
    }
    DijkNode(int u, int d):u(u), d(d){}
};

class Solution {
private:
    // 把图的输入 边,转为 邻接表
    vector<vector<pair<int, int> > > E;
    // Dijkstra 计算每个节点当前得到的最短路
    vector<int> dist;
    // 访问标志,在 Dijkstra求最短路, 和 Dfs搜索路径时需要用到
    vector<bool> vis;
    
    const int MODE = 1000000007;

public:
    int countRestrictedPaths(int n, vector<vector<int> >& edges) {
        
        dist = vector<int>(n+1, INT_MAX);
        vis = vector<bool>(n+1, false);
        // E 邻接表设为n+1 的大小
        E = vector<vector<pair<int, int> > >(n+1);

        for(vector<vector<int> >::iterator it = edges.begin(); it != edges.end(); it++) {
            E[(*it)[0]].push_back(make_pair((*it)[1], (*it)[2]));
            E[(*it)[1]].push_back(make_pair((*it)[0], (*it)[2]));
        }

        // 初始化一个优先队列, 优先队列默认是 大顶堆,可以用小顶堆替代。
        priority_queue<DijkNode, vector<DijkNode>, greater<DijkNode> > queue;

        // 进行Dijkstra 计算每个节点到 n的最短路径
        dist[n] = 0;
        queue.push(DijkNode(n, 0));
        while (!queue.empty()) {
            int u = queue.top().u; queue.pop();
            if (vis[u]) continue; // 队列里有多个相同 u 的DijkNode,d最小的那个已经访问过了。
            // 标记
            vis[u] = true;
            for(vector<pair<int, int> >::iterator it = E[u].begin(); it != E[u].end(); it++) {
                pair<int, int> edge = *it;
                if (vis[edge.first]) continue;
                if (dist[edge.first] > dist[u] + edge.second) {
                    dist[edge.first] = dist[u] + edge.second;
                    queue.push(DijkNode(edge.first, dist[edge.first]));
                }
            }
        }
        
        // 搜索路径, 记忆化搜索
        vector<ll> memo(n+1, -1);
        fill(vis.begin(), vis.end(), false);
        ll ans = memoSearch(1, n, memo);
        return (int)ans;
    }

    ll memoSearch (int u, int n, vector<ll> &memo)
    {
        if (memo[u] != -1)
            return memo[u];
        if (u == n) 
            return 1;
        
        vis[u] = true;
        ll pathcount = 0;
        for(vector<pair<int, int> >::iterator it = E[u].begin(); it != E[u].end(); it++) {
            pair<int, int> edge = *it;
            if (!vis[edge.first] && dist[u] > dist[edge.first]) {
                // 因为只找有限路径,dfs时有条件:dist[u] > dist[edge.first],
                // 所以可以用记忆化搜索,查找时不会重复
                pathcount = (pathcount + memoSearch(edge.first, n, memo)) % MODE;
            }
        }
        vis[u] = false;
        memo[u] = pathcount;
        return pathcount;
    }
};

229. 求众数 II 10月22日

题目链接

摩尔投票:抵消阶段 + 计数阶段

class Solution {
public:
    vector<int> majorityElement(vector<int>& nums) {
        int n = nums.size();
        int major1 = 0;
        int major2 = 0;
        int vote1 = 0, vote2 = 0;
        int limit = n / 3;
        // 抵消阶段
        for(auto num: nums) {
            if (vote1 > 0 && num == major1) {
                vote1++;
            } else if (vote2 > 0 && num == major2){
                vote2++;
            } else if (vote1 == 0) {
                major1 = num;
                vote1++;
            } else if (vote2 == 0) {
                major2 = num;
                vote2++;
            } else {
                vote1--;
                vote2--;
            }
            
        }
        
        // 计数阶段
        vector<int> ans;
        int count1 = 0, count2 = 0;

        for (auto num : nums) {
            if (num == major1)
                count1++;
            else if (num == major2)
                count2++;
        }
        if (count1 > limit)
            ans.push_back(major1);
        if (count2 > limit)
            ans.push_back(major2);
        
        return ans;
    }
};

496. 下一个更大元素 I 10月26日

题目链接

单调栈

class Solution {
public:
    vector<int> nextGreaterElement(vector<int>& nums1, vector<int>& nums2) {
        map<int,int> dict;
        
        int n2 = nums2.size();
        int n1 = nums1.size();
        stack<int> stc;
        vector<int> ans(n1);
        
        for(int num : nums2) {
            while(!stc.empty() && stc.top() < num) {
                dict[stc.top()] = num;
                stc.pop();
            }
            stc.push(num);
        }
        while(!stc.empty()) {
            dict[stc.top()] = -1;stc.pop();
        }
        for(int i=0; i<n1; i++) {
            ans[i] = dict[nums1[i]];
        }
        return ans;
    }
};

503. 下一个更大元素 II 10月26日

题目链接

数组里有重复数字了,所以入栈内容的应该是下标位置。

class Solution {
public:
    vector<int> nextGreaterElements(vector<int>& nums) {
        int n = nums.size();
        stack<int> stc;
        vector<int> ans(n, -1);
        for(int i=0; i<2*n-1; i++) {
            while (!stc.empty() && nums[stc.top()] < nums[i%n])