题目:给你一个二维整数数组 envelopes
,其中 envelopes[i] = [wi, hi]
,表示第 i
个信封的宽度和高度。
当另一个信封的宽度和高度都比这个信封大的时候,这个信封就可以放进另一个信封里,如同俄罗斯套娃一样。
请计算 最多能有多少个 信封能组成一组“俄罗斯套娃”信封(即可以把一个信封放到另一个信封里面)。
注意:不允许旋转信封。
示例 1:
输入:envelopes = [[5,4],[6,4],[6,7],[2,3]]
输出:3
解释:最多信封的个数为 3, 组合为:
[2,3] => [5,4] => [6,7]。
示例 2:
输入:envelopes = [[1,1],[1,1],[1,1]] 输出:1j
解法1:常规解法->动态规划
步骤:1.状态表示 2.状态转移方程 3.初始化 4.填表顺序 5.返回值
import java.util.Arrays;public class Solution {public int maxEnvelopes(int[][] e) {Arrays.sort(e,(v1,v2)->{return v1[0] - v2[0];});int n=e.length;int[]dp=new int[n];int ret=1;for (int i=1;i<n;i++){dp[i]=1;for (int j=0;j<i;j++){if (e[i][0]>e[j][0]&&e[i][1]>e[j][1]){dp[i]=Math.max(dp[i],dp[i]+1);}}ret=Math.max(ret,dp[i]);}return ret;}public static void main(String[] args) {Solution solution=new Solution();int [][] e={{5,4},{6,4},{6,7},{2,3}};System.out.println(solution.maxEnvelopes(e));}
}
解法2:重写排序+贪心+二分
重写排序:1.当左端点不同的时候,按左端点从小到大的顺序排列;
2.当左端点相同的时候,按右端点从大到小的顺序排列;
import java.util.ArrayList;
import java.util.Arrays;public class Solution2 {public int maxEnvelopes(int[][] e){Arrays.sort(e,(v1,v2)->{return v1[0]!=v2[0]?v1[0]-v2[0]:v2[1]-v1[1];});ArrayList<Integer>ret=new ArrayList<>();ret.add(e[0][1]);for (int i=1;i<e.length;i++){int b=e[i][1];if(b>ret.get(ret.size()-1)){ret.add(b);}else {int left=0,right=ret.size()-1;while(left>right){int mid=(left+right)/2;if(ret.get(mid)>=b)right=mid;else left=mid+1;}ret.set(left,b);}}return ret.size();}public static void main(String[] args) {Solution solution1=new Solution();int[][] e={{5,4},{6,4},{6,7},{2,3}};System.out.println(solution1.maxEnvelopes(e));}
}