首頁 >Java >java教程 >使用java實作LIS演算法,出操隊形的問題

使用java實作LIS演算法,出操隊形的問題

高洛峰
高洛峰原創
2017-01-18 14:38:041564瀏覽

假設有序列:2,1,3,5,求一個最長上升子序列為2,3,5或1,3,5,長度皆為3。

LIS演算法的想法是:

設存在序列a。

① 如果只有一個元素,那麼最長上升子序列的長度為1;

② 如果有兩個元素,那麼如果a[1]>a[0],則最長上升子序列的長度為2 ,a[1]為此最長上升子序列的最後一個元素;若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個整數:代表學生的身高(cm)(身高為不高於200的正整數)。

輸出:

對應每個測試案例,輸出需要抽出的最少學生人數。

樣本例輸入:

6

100 154 167 159 132 105

5

152 152 152 152 150%

在用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算法,出操隊形的問題的全部內容了,希望對大家有所幫助,多多支持PHP中文網~

更多使用java實現LIS算法,出操隊形的問題相關文章請關注PHP中文網!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn