假设有序列:2,1,3,5,求一个最长上升子序列就是2,3,5或者1,3,5,长度都为3。
LIS算法的思想是:
设存在序列a。
① 如果只有一个元素,那么最长上升子序列的长度为1;
② 如果有两个元素,那么如果a[1]>a[0],则最长上升子序列的长度为2,a[1]为该最长上升子序列的最后一个元素;若a[1]<a[0],则最长上升子序列的长度为1,a[0]和a[1]均为 其最长上升子序列的最后一个元素。
③ 如果由三个元素,那么如果a[2]>a[0],a[2]>a[1],则a[2]可以作为a[0]或者a[1]所在最长上升子序列的最后一个元素。那选择哪一个序列就要看a[0],a[1]哪个所在的序列要更长。
④ 扩展到n个元素,就是看以a[n]为最后一个元素的最长上升子序列的长度是多少。
定义两个数组,一个是a,一个是b。
a存放原始数据,b[i]存放的是以a[i]结尾的最长上升子序列的长度。
代码如下:
class Lmax{ html" target="_blank">public static void Lmax(int[] a,int[] b){ b[0]=1; for(int i=1;i<a.length;i++){ int countmax=0; for(int j=0;j<i;j++){ if(a[i]>a[j]&&b[j]>countmax){ countmax=b[j]; //记录下元素数值比a[i]小的但是对应子序列最长的子序列长度 } } b[i]=countmax+1; //a[i]对应的最长子序列长度是 } } }
二、出操队形
题目描述:
在 读高中的时候,每天早上学校都要组织全校的师生进行跑步来锻炼身体,每当出操令吹响时,大家就开始往楼下跑了,然后身高矮的排在队伍的前面,身高较高的就 要排在队尾。突然,有一天出操负责人想了一个主意,想要变换一下队形,就是当大家都从楼上跑下来后,所有的学生都随机地占在一排,然后出操负责人从队伍中 抽取出一部分学生,使得队伍中剩余的学生的身高从前往后看,是一个先升高后下降的“山峰”形状。据说这样的形状能够给大家带来好运,祝愿大家在学习的道路 上勇攀高峰。(注,山峰只有一边也符合条件,如1,1、2,2、1均符合条件)
输入:
输入可能包含多个测试样例。
对于每个测试案例,输入的第一行是一个整数n(1<=n<=1000000):代表将要输入的学生个数。
输入的第二行包括n个整数:代表学生的身高(cm)(身高为不高于200的正整数)。
输出:
对应每个测试案例,输出需要抽出的最少学生人数。
样例输入:
6
100 154 167 159 132 105
5
152 152 152 152 152
样例输出:
0
4
在用LIS来解这道题的时候,可以这样考虑:
首先从前向后用LIS求一遍以每一个元素结尾的最长上升子序列的长度,然后将数组逆序,再用LIS求一遍以每一个元素结尾的最长上升子序列的长度。
得到两个数组b1,b2。
b1,b2对应相加再减去重复的一个,就是最长的'山峰'。
public class peak { public static void main (String[] args) { int n; int re; do{ Scanner in = new Scanner(System.in); n = in.nextInt(); }while(n<0||n>100000); int []a = new int[n]; //原始数组 int []ar = new int[n]; //逆序数组 Scanner in = new Scanner(System.in); for(int i=0;i<n;i++){ a[i]=in.nextInt(); } int[] b1 = new int[n]; @SuppressWarnings("unused") int[] b2 = new int[n]; Lmax.Lmax(a, b1); ar=reverse.reverse(a); Lmax.Lmax(ar, b2); //求解逆序数组的最长上升子序列 b2=reverse.reverse(b2); //将逆序数组的最长上升子序列逆序以便和原始数组的最长上升子序列对应相加 re = result.result(b1, b2); System.out.print(re); } }<br><br><br><br>
class result{ public static int result(int[] a,int[] b){ int max=0; int[] c = new int[a.length]; for(int i=0;i<a.length;i++){ c[i]=a[i]+b[i]; } Arrays.sort(c); max=c[c.length-1]-1; //对应相加最长的再减去重复的一个人 return a.length-max; } }
以上就是小编为大家带来的使用java实现LIS算法,出操队形的问题的全部内容了,希望对大家有所帮助,多多支持小牛知识库~
用java实现以下场景,有100条商品,每个商品的价格0到1000元不等 商品数据: 例如200元 加价之后就是240元 利润是40元 首先把0到300元的商品的加价30%,301到500的加价20%,501到1000元商品的加价10%, 然后0到300元的商品订单有300条,301-500的商品订单有400条,501-1000的商品订单有300条, 最后计算他的总利润有多少,最好写出算法
我正在使用优先级队列实现Dijkstra的算法,我想要一个函数从堆中删除一个元素,但我只能从Dijkstra的主节点索引中向它发送顶点索引,我找不到它在堆上的位置,我负担不起进行二进制搜索。有什么想法吗?
2)Java的内置使用了两个锁:takeLock和putLock,并分别用在put()和take()中,我看到间隔队列是一个链表,不是线程安全的,那怎么行呢?
本文向大家介绍LIS 最长递增子序列 Java的简单实现,包括了LIS 最长递增子序列 Java的简单实现的使用技巧和注意事项,需要的朋友参考一下 今天遇到了一个求最长递增子序列的问题,看了之后就尝试着用Java实现了一下,关于什么是最长递增子序列,这里就不在赘述,可以百度或者Google之,以下为实现的代码: 说明:本段代码实现的功能为 (1)随机生成一个有10个元素的数组,然后输出它的最长递增
问题内容: 抱歉,下面是我在这里遇到的一个问题:在这里,我试图运行此方法以从双面队列(双端队列)中删除通用值(EltType),但是我一直遇到错误,我两次调用insertFirst ,然后将值“ 3”插入数组两次,然后,当我运行removeFirst时,它将打印出“ 3”一次,然后打印出“ Null”。有人可以帮我吗? 谢谢 :) 问题答案: 明显的问题是它永远不会改变。 将永远返回。现在,让我们
我正在寻找一种通用模式来分解一些常见的代码:我需要在具有不同算术属性的类之间实现代数加法和减法。一个典型的例子是可以用秒、小时和分钟表示的间隔,我用一个具有三个int属性的类实现了它。 例如,如果我想减去0秒、0分钟、1小时的周期,减去0秒、30分钟、0小时的周期,我不想获得0秒、-30分钟、1小时的周期。 我需要编码两个时间间隔之间的加减法,有没有通用的模式来编码这个代数?我应该寻找不同的代表吗