首页 > 编程语言 > 详细

luogu P5414 [YNOI2019]排序

时间:2019-12-09 22:55:55      阅读:105      评论:0      收藏:0      [点我收藏+]

题目描述

对于一个数列{7, 1, 2, 3}进行排序,我们可以把7 从头移动到尾。但是这个操作的成本是7,并不是最佳的。最佳的排序方式是将连续的1、2、3 移动到7 的前面。这样的话,总的操作成本就是1+2+3=6,比之前的成本7 要小。

你的任务是,对于一个给定的数列,输出对这个数列进行排序的最小成本。

输入格式

输入文件名为sort.in。

每个输入文件包含多组数据。

输入文件的第一行,包含一个正整数T,代表该输入文件中所含的数据组数。

接下来是T组数据,每组数据的格式如下:

每组数据包含2 行;

第一行包含一个正整数n,代表数列中元素的个数,其中0 < n ≤ 102;

第二行包含n个整数,两个数之间以一个空格隔开,代表数列中的元素ki,其中?107-10^{7}?107 ≤ ki ≤ 10710^{7}107。

输出格式

输出文件名为sort.out。

输出文件包含T行,分别对应T组数据的答案,即对数列进行排序的最小成本。

说明/提示

对于60%的数据:0 < n ≤ 60,?107-10^{7}?107 ≤ ki ≤ 10710^{7}107

对于80%的数据:0 < n ≤ 80, ?107-10^{7}?107 ≤ ki ≤ 10710^{7}107

对于100%的数据:0 < n ≤ 102,?107-10^{7}?107 ≤ ki ≤ 10710^{7}10


这道题就是移动几个数字使得数列单调上升,每次移动可以移动到任意位置,花费就是移动数字的大小。

看到这个题之后,我们可以得到 得不到的洗洗睡吧 :最优解不可能把同一个数字移动2次及以上。既然我们可以一次把它移到正确的位置上,那么这个数字就可以理解为直接被移走了。

那么问题就变成了:删除几个数字,使得数列单调递增,所求的是移走数字的最小和。 使得删除的和最小,就意味着留下的和最大。

那么问题就变成了:保留几个单调递增的数字,使得和最大。

#include<map>
#include<ctime>
#include<cstdlib>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
inline int read(){
    int x=0; char c=getchar();
    while(c<'0'||c>'9')c=getchar();
    while('0'<=c&&c<='9'){ x=(x<<3)+(x<<1)+(c^48); c=getchar();}
    return x;
}
const int N=205;
int a[N],dp[N];
signed main(){
    register int i,j; int n;
    for(int T=read(),sum=0,ans=0;T;T--,sum=0,ans=0){
        n=read(); for(i=1;i<=n;i++){
            a[i]=read(),sum+=a[i];
            for(j=1;j<i;j++)if(a[j]<=a[i]&&dp[i]<dp[j])dp[i]=dp[j];
            dp[i]+=a[i];
        }
        for(i=1;i<=n;i++)ans=max(ans,dp[i]);
        printf("%d\n",sum-ans);
    }
}

luogu P5414 [YNOI2019]排序

原文:https://www.cnblogs.com/naruto-mzx/p/12013785.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!