zoukankan      html  css  js  c++  java
  • 九度1500-出操队形

    题目描述:
    在读高中的时候,每天早上学校都要组织全校的师生进行跑步来锻炼身体,每当出操令吹响时,大家就开始往楼下跑了,然后身高矮的排在队伍的前面,身高较高的就要排在队尾。突然,有一天出操负责人想了一个主意,想要变换一下队形,就是当大家都从楼上跑下来后,所有的学生都随机地占在一排,然后出操负责人从队伍中抽取出一部分学生,使得队伍中剩余的学生的身高从前往后看,是一个先升高后下降的“山峰”形状。据说这样的形状能够给大家带来好运,祝愿大家在学习的道路上勇攀高峰。(注,山峰只有一边也符合条件,如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
    分析:
    这里应使用O(nlogn)的算法。先求总体递增子序列的长度,再求递减子序列的长度。这里为了复用LIS(最长上升序列长度)函数,把原来的数组逆序一下,就相当于求递减子序列的长度了。
    参考代码如下:
     
     1 #include<iostream>
     2 #include<algorithm>
     3 #include<stdlib.h>
     4 
     5 using namespace std;
     6 
     7 int LongestGrowingNumberLength( int* arr,int length,int start)//从start处开始递增数组的最大长度
     8 {
     9     if(arr==NULL||length<=0||start<0) return 0;
    10     int* begin=arr+start;
    11     int* next=begin+1;
    12     int count=1;
    13     while(next<arr+length)
    14     {
    15         if(*begin==*next)
    16         {
    17          next++;
    18          begin++;
    19         }
    20         if(*begin<*next)
    21         {
    22             *begin=*next;
    23             next++;
    24             count++;
    25         }
    26         else next++;
    27     }
    28     return count;
    29 }
    30 
    31 int GrowingNumberLengthCore(int* arr,int length,int start,int end)//最大正序递增长度
    32 {
    33     if(arr==NULL||length<0||start<0||end>length-1) return -1;
    34     int MaxValue=LongestGrowingNumberLength(arr,length,0);
    35     for(int i=1;i<end;i++)
    36     {   
    37        int L=LongestGrowingNumberLength(arr,length,i);
    38         if(MaxValue< L)
    39         {MaxValue=L;}
    40     }
    41     return MaxValue;
    42 }
    43 
    44 void main()
    45 {
    46     int n;
    47     cin>>n;
    48     int* array=new int[n+1];
    49     int* resever=new int[n+1];
    50     for(int i=0;i<n;i++)
    51     {cin>>array[i];}
    52     for(i=0;i<n;i++)
    53     {resever[i]=array[n-1-i];}
    54     int grow=GrowingNumberLengthCore(array,n,0,n-1);//最大正序递增长度
    55     int down=GrowingNumberLengthCore(resever,n,0,n-1);//最大正序递减长度也就是最大逆序递增长度
    56     cout<<grow<<endl;
    57     cout<<down<<endl;
    58     cout<<(n+1-grow-down)<<endl;//最少删除的元素数
    59 }
  • 相关阅读:
    ubuntu重复登录问题
    CUDA相关问题
    pytorch安装(使用pip3装到conda环境下)
    ubuntu16.04 anaconda的安装和卸载
    vscode插件安装失败的解决方案
    使用ffmpeg进行视频截图
    Spring加载早期获取BasePackage
    chrome最耐看的主题
    针对MySQL的MVCC多版本并发控制的一些总结
    docker创建mysql容器,并挂载数据+配置
  • 原文地址:https://www.cnblogs.com/wxdjss/p/5697400.html
Copyright © 2011-2022 走看看