赞
踩
力扣第300题和蓝桥杯合唱队
都是采用动态规划
来回忆一下动态规划步骤:前者的结果对后者有影响,或者最优,最短等字眼第一感觉就是用动态规划
1.定义dp数组,初始化dp数组
2.模拟枚举前几个得出 动态转移方程
3.返回最终的解(注意当输入为0 ,1这些边界问题应该返回的值是否与dp数组一致。)
力扣300 https://leetcode-cn.com/problems/longest-increasing-subsequence/
给你一个整数数组 nums ,找到其中最长严格递增子序列的长度。
子序列 是由数组派生而来的序列,删除(或不删除)数组中的元素而不改变其余元素的顺序。例如,[3,6,2,7] 是数组 [0,3,1,6,2,2,7] 的子序列。
class Solution { public int lengthOfLIS(int[] nums) { int len = nums.length; int[] dp = new int[len]; dp[0] = 1;//初始化 int maxans = 1; for (int i = 1; i < nums.length; i++) { dp[i] = 1;//初始化dp无论哪一个位置至少是1 for (int j = 0; j < i; j++) { if (nums[i] > nums[j]) {//只有nums[i] > nums[j] 才会出现递增的情况 /** * 动态转移方程:dp[i]代表当前位置有的值 * dp[j]+1代表 在遍历过程中,在j到i区间里 因为nums[i] > nums[j] * 推出了 多了一个递增的数 所以dp[i]可能等于dp[j]+1 * 显然可能等于就是取dp[i]和dp[j+1]之间的大者 */ dp[i] = Math.max(dp[i], dp[j] + 1); } } maxans = Math.max(maxans, dp[i]);//计算dp中的最大值 } return maxans; } }
题目描述
NN 位同学站成一排,音乐老师要请其中的 (N-K)(N−K) 位同学出列,使得剩下的 KK 位同学排成合唱队形。
合唱队形是指这样的一种队形:设 KK 位同学从左到右依次编号为 1-K,他们的身高分别为T1-TK , 则他们的身高满足 中间高两边低,就像^形状一样。
你的任务是,已知所有 NN 位同学的身高,计算最少需要几位同学出列,可以使得剩下的同学排成合唱队形。
思路:就是求剩下最多人
从左到右求递增序列的
从右到左求单调递增序列
二者和最大就是剩下最多
public static void main(String[] args) { Scanner scan = new Scanner(System.in); //在此输入您的代码... int len=scan.nextInt(); int [] arr=new int[len]; for (int i = 0; i < len; i++) { arr[i]=scan.nextInt(); } int[] dp=new int[len];//存储最大递增子序列 int[] dp2=new int[len];//最大递减子序列 //力扣300的求法 dp[0]=dp2[0]=1; for (int i = 1; i < len; i++) { dp[i]=1; for (int j = 0; j < i; j++) { if (arr[i]>arr[j]){ dp[i] = Math.max(dp[i], dp[j] + 1); } } } //反向遍历求法和变成递增子序列,实质是递减 for (int i = len-1; i >= 0; i--) { dp2[i]=1; for (int k = len-1; k > i; k--) { if (arr[i]>arr[k]){ dp2[i] = Math.max(dp2[i], dp2[k] + 1); } } } //二者中取和最大,但是中间值被算了两次所以减1 int max=0; for (int i = 0; i < len; i++) { max=Math.max(max,dp[i]+dp2[i]-1); } System.out.println(len-max); }
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。