zoukankan      html  css  js  c++  java
  • HWOJ-合唱队

    计算最少出列多少位同学,使得剩下的同学排成合唱队形

    说明:

    N位同学站成一排,音乐老师要请其中的(N-K)位同学出列,使得剩下的K位同学排成合唱队形。 
    合唱队形是指这样的一种队形:设K位同学从左到右依次编号为1,2…,K,他们的身高分别为T1,T2,…,TK,   则他们的身高满足存在i(1<=i<=K)使得Ti<T2<......<Ti-1<Ti>Ti+1>......>TK。 
         你的任务是,已知所有N位同学的身高,计算最少需要几位同学出列,可以使得剩下的同学排成合唱队形。 

    输入

    整数N

    一行整数,空格隔开,N位同学身高

     

    输出

    最少需要几位同学出列

    样例输入 8 186 186 150 200 160 130 197 200
    样例输出 4

    这个问题实际就是在求一个最长递增子序列,和最长递减子序列的问题,对应求和找到最大的那个temp=arrayLenUp[i]+arrayLenDown[N-1-i];,即是合唱队的长度-1。

    Java代码:通过

    import java.util.Scanner;
    
    public class Main {
    
        public static void main(String[] args){
    
            Scanner scanner=new Scanner(System.in);
            System.out.println("请输入一个整数:");
            int N=scanner.nextInt();
            int[] height=new int[N];
            for (int i = 0; i < N; i++) {
                height[i]=scanner.nextInt();
            }
            Main main=new Main();
            int[] arrayLenUp=main.getLISUp(height);
            for(int i:arrayLenUp)
                System.out.print(i+",");
            System.out.println();
            int[] arrayLenDown=main.getLISDown(height);
            for(int i:arrayLenDown)
                System.out.print(i+",");
            System.out.println();
            int total=2;
            int temp;
            for (int i = 0; i < N; i++) { //对应求和找到最大的那个
                temp=arrayLenUp[i]+arrayLenDown[N-1-i];
                if (temp>total) {
                    total=temp;
                }
            }
            System.out.println((N-total+1)); //输出最终结果
            scanner.close();
    
        }
    
       
    
        public int binarySearchPosition(int arrayOut[],int left,int right,int key){ //二分查找要替换的位置
    
            int mid;
    
            if (arrayOut[right]<key) {
                return right+1;
            }else {
                while(left<right){
                    mid=(left+right)>>1;
                    if (arrayOut[mid]<key) {
                        left=mid+1;
                    }else {
                        right=mid;
                    }       
                }
                    return left;
            }
    
        }
    
        public int[] getLISUp(int[] arrayIn){ //获取最长递增子序列并把它们保存在数组arrayLen中
    
            int len=1;
            int position;
            int[] arrayOut=new int[arrayIn.length+1];
            arrayOut[1]=arrayIn[0];
            int[] arrayLen=new int[arrayIn.length];
            arrayLen[0]=1;
            for (int i = 1; i < arrayIn.length; i++) {
                position=binarySearchPosition(arrayOut, 1, len, arrayIn[i]);
                arrayOut[position]=arrayIn[i];
                if (position>len) {
                    len=position;
                }
                arrayLen[i]=position;
            }
            return arrayLen;
        }
    
        public int[] getLISDown(int[] arrayIn){ ////获取最长递减子序列并把它们保存在数组arrayLen中
            int[] arrayReverse=new int[arrayIn.length];
            int[] arrayLen=new int[arrayIn.length];
            for (int i = 0; i < arrayReverse.length; i++) { //将最长递减子序列问题转换为最长递增子序列问题
                arrayReverse[i]=arrayIn[arrayIn.length-1-i];
            }
            arrayLen=getLISUp(arrayReverse);
            return arrayLen;
        }
    
    }

    C代码:没有通过

    #include "iostream"  
    
    #include "stdio.h"  
    
    #include "math.h"  
    
    #include "vector"  
    
    #include "queue"  
    
    #include "memory.h"  
    
    #include "algorithm"  
    
    #include "string"  
    
    using namespace std;  
    
      
    
    int inc1[200],inc2[200],a[200];  
    
    //inc1-->longest increase array from head to tail  
    
    //inc2-->longest increase array from tail to head  
    
      
    
    int main()  
    
    {  
    
        int n;  
    
        while(scanf("%d",&n)!=EOF)  
    
        {  
    
            int ans=0,i,j;  
    
            for(i=1;i<=n;i++)  
    
                scanf("%d",&a[i]);  
    
            inc1[1]=1;  
    
            for(i=2;i<=n;i++)  
    
            {  
    
                inc1[i]=1;  
    
                for(j=1;j<i;j++)  
    
                    if(a[i]>a[j]&&inc1[j]+1>inc1[i])  
    
                        inc1[i]=inc1[j]+1;  
    
            }  
    
            inc2[n]=1;  
    
            for(i=n-1;i>=1;i--)  
    
            {  
    
                inc2[i]=1;  
    
                for(j=n;j>i;j--)  
    
                    if(a[j]<a[i]&&inc2[j]+1>inc2[i])  
    
                        inc2[i]=inc2[j]+1;  
    
            }  
    
            for(i=1;i<=n;i++)  
    
                if(inc1[i]+inc2[i]-1>ans)   
    
                    ans=inc1[i]+inc2[i]-1;  
    
            printf("%d
    ",n-ans);  
    
        }  
    
        return 0;  
    
    }  
  • 相关阅读:
    美化单选按钮
    url的处理
    时间戳处理
    模板渲染引擎手册
    Web前端工具——Live Reload – Chrome扩展
    属性(Properties)和字段在C#中的关系
    MVC3中的Model
    使用System.Web.Mvc.Ajax
    javascript之DOM操作
    Javascript之BOM(window对象)
  • 原文地址:https://www.cnblogs.com/hupp/p/4761631.html
Copyright © 2011-2022 走看看