码迷,mamicode.com
首页 > 其他好文 > 详细

[luogu2501 HAOI2006] 数字序列 (递推LIS)

时间:2018-06-09 16:46:32      阅读:198      评论:0      收藏:0      [点我收藏+]

标签:递推   code   har   abs   eid   绝对值   line   改变   find   

题目描述

现在我们有一个长度为n的整数序列A。但是它太不好看了,于是我们希望把它变成一个单调严格上升的序列。但是不希望改变过多的数,也不希望改变的幅度太大。

输入输出格式

输入格式:
第一行包含一个数n,接下来n个整数按顺序描述每一项的键值。

输出格式:
第一行一个整数表示最少需要改变多少个数。

第二行一个整数,表示在改变的数最少的情况下,每个数改变的绝对值之和的最小值。

输入输出样例

输入样例#1:
4
5 2 3 5
输出样例#1:
1
4

说明

【数据范围】

90%的数据n<=6000。

100%的数据n<=35000。

保证所有数列是随机的。

题解

只会做第一问 比较简单求[a[i]-i]最长不降
第二问 膜拜题解

code:

//By Menteur_Hxy
#include <map>
#include <set>
#include <cmath>
#include <queue>
#include <cstdio>
#include <string>
#include <vector>
#include <cstring>
#include <iostream>
#include <algorithm>
#define LL long long
#define M(a,b) memset(a,(b),sizeof(a))
#define F(i,a,b) for(register int i=(a);i<=(b);i++)
#define C(i,a,b) for(register int i=(b);i>=(a);i--)
#define E(i,u) for(register int i=head[u];i;i=nex[i])
using namespace std;

inline LL rd() {
    LL x=0,fla=1; char c=‘ ‘;
    while(c>‘9‘|| c<‘0‘) {if(c==‘-‘) fla=-fla; c=getchar();}
    while(c<=‘9‘ && c>=‘0‘) x=x*10+c-‘0‘,c=getchar();
    return x*fla;
}

inline void out(LL x){
    int a[30],wei=0;
    if(x<0) putchar(‘-‘),x=-x;
    for(;x;x/=10) a[++wei]=x%10;
    if(wei==0){ puts("0"); return;}
    for(int j=wei;j>=1;--j) putchar(‘0‘+a[j]);
    putchar(\n);
}

const int N=35010;
const int INF=0x3f3f3f3f;
int n,tp,cnt,L;
int b[N],f[N],head[N],to[N],nex[N],mn[N];
LL g[N],s1[N],s2[N];

void add(int x,int y) {
    nex[++cnt]=head[x],to[cnt]=y,head[x]=cnt;
}

int find(int x) { int l=1,r=L,t=0;
    while(l<=r) { int mid=(l+r)>>1;
        if(mn[mid]<=x) t=mid,l=mid+1;
        else r=mid-1;
    }
    return t;
}

void dp() {
    M(mn,127);
    mn[0]=-INF;
    F(i,1,n) { int t=find(b[i]);
        f[i]=t+1;
        L=max(L,t+1);
        mn[t+1]=min(mn[t+1],b[i]);
    }
}

int main() {
    n=rd();
    F(i,1,n) b[i]=rd()-i; b[++n]=INF;
    dp();
    C(i,0,n) add(f[i],i),g[i]=(LL)INF;
    g[0]=0;b[0]=-INF;
    F(x,1,n) E(i,f[x]-1) { int p=to[i];
        if(p>x) break;
        if(b[p]>b[x]) continue;
        F(j,p,x) s1[j]=abs(b[p]-b[j]),s2[j]=abs(b[x]-b[j]);
        F(j,p+1,x) s1[j]+=s1[j-1],s2[j]+=s2[j-1];
        F(j,p,x-1) g[x]=min(g[x],g[p]+s1[j]-s1[p]+s2[x]-s2[j]);
    }
    out(n-f[n]);out(g[n]);
    return 0;
}  

[luogu2501 HAOI2006] 数字序列 (递推LIS)

标签:递推   code   har   abs   eid   绝对值   line   改变   find   

原文地址:https://www.cnblogs.com/Menteur-Hxy/p/9159734.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!