星空网 > 软件开发 > Java

java实现LIS算法,出操队形问题

假设有序列: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{		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);	}}




 

 

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

*特别声明:以上内容来自于网络收集,著作权属原作者所有,如有侵权,请联系我们: admin#shaoqun.com (#换成@)。

除了转化率还有什么影响店铺流量?:https://www.ikjzd.com/articles/130217
大批亚马逊卖家旺季前被收割,60多万被扣,杀伤力太强:https://www.ikjzd.com/articles/130219
阿里国际站店铺诊断攻略!(深度分析):https://www.ikjzd.com/articles/130220
万圣节旺季在即,送你一份远离侵权风险指南!:https://www.ikjzd.com/articles/130222
Lazada新商家成长能力之产品优化:https://www.ikjzd.com/articles/130223
跨境卖家选品时应注意什么?都有哪些策略技巧?:https://www.ikjzd.com/articles/130224
船底顶攻略(船底顶最详细攻略):https://www.vstour.cn/a/411239.html
川藏线自驾游要怎么走才比较划算呢?:https://www.vstour.cn/a/411240.html
相关文章
我的浏览记录
最新相关资讯
海外公司注册 | 跨境电商服务平台 | 深圳旅行社 | 东南亚物流