记录编号 285131 评测结果 AAAAAAAAAA
题目名称 [网络流24题] 最长递增子序列 最终得分 100
用户昵称 GravatarTenderRun 是否通过 通过
代码语言 C++ 运行时间 0.014 s
提交时间 2016-07-20 22:33:41 内存使用 14.43 MiB
显示代码纯文本
#include <iostream>
#include <cstring>
#include <cstdio>
#include <queue>
using namespace std;
const int maxn=100010;
const int maxm=1000010;
const int INF=1000000000;
int cnt,fir[maxn],to[maxm],nxt[maxm],cap[maxm];
int dis[maxn],gap[maxn],path[maxn],fron[maxn];
queue<int>q;

struct Net_Flow{
	int tot;
	void Init(int tot_){
		memset(dis,0,sizeof(dis));
		memset(gap,0,sizeof(gap));
		memset(fir,0,sizeof(fir));
		cnt=1;tot=tot_;
	}
	void add(int a,int b,int c){
		nxt[++cnt]=fir[a];
		cap[cnt]=c;
		fir[a]=cnt;
		to[cnt]=b;
	}
	void addedge(int a,int b,int c){
		add(a,b,c);add(b,a,0);
	}
	bool BFS(int S,int T){
		dis[T]=1;q.push(T);
		while(!q.empty()){
			int x=q.front();q.pop();
			for(int i=fir[x];i;i=nxt[i])
				if(!dis[to[i]]){
					dis[to[i]]=dis[x]+1;
					q.push(to[i]);
				}
		}
		return dis[S];
	}
	int Max_Flow(int S,int T){
		if(!BFS(S,T))return 0;
		for(int i=0;i<tot;i++)fron[i]=fir[i];
		for(int i=0;i<tot;i++)gap[dis[i]]+=1;
		int ret=0,p=S,f,Min;
		while(dis[S]<=tot){
			if(p==T){
				f=INF;
				while(p!=S){
					f=min(f,cap[path[p]]);
					p=to[path[p]^1];
				}ret+=f;p=T;
				while(p!=S){
					cap[path[p]]-=f;
					cap[path[p]^1]+=f;
					p=to[path[p]^1];
				}
			}
			
			for(int &i=fron[p];i;i=nxt[i])
				if(cap[i]&&dis[to[i]]==dis[p]-1)
					{path[p=to[i]]=i;break;}
			
			if(!fron[p]){Min=tot;
				if(--gap[dis[p]]==0)break;
				for(int i=fir[p];i;i=nxt[i])
					if(cap[i])Min=min(Min,dis[to[i]]);
				gap[dis[p]=Min+1]+=1;fron[p]=fir[p];
				if(p!=S)p=to[path[p]^1];	
			}		
		}
		return ret;
	}
}ISAP;

int a[maxn];
int f[maxn];
int main(){
	freopen("alis.in","r",stdin);
	freopen("alis.out","w",stdout);
	int n,S,T,ans=1;
	scanf("%d",&n);S=0;
	ISAP.Init(n+2);T=n+1;
	for(int i=1;i<=n;i++)
		scanf("%d",&a[i]);	
	
	for(int i=1;i<=n;i++){
		f[i]=1;
		for(int j=i-1;j>=1;j--)
			if(a[i]>=a[j])f[i]=max(f[i],f[j]+1);
		ans=max(f[i],ans);
	}
	printf("%d\n",ans);
	
	for(int i=1;i<=n;i++){
		if(f[i]==1)ISAP.addedge(S,i,1);
		if(f[i]==ans)ISAP.addedge(i,T,1);
		for(int j=i-1;j;j--)
			if(f[j]==f[i]-1&&a[i]>=a[j])
				ISAP.addedge(j,i,1);
	}
	printf("%d\n",ISAP.Max_Flow(S,T));
	ISAP.Init(n+2);S=0;T=n+1;
	for(int i=1;i<=n;i++){
		if(f[i]==1)ISAP.addedge(S,i,1);
		if(f[i]==ans)ISAP.addedge(i,T,1);
		for(int j=i-1;j;j--)
			if(f[j]==f[i]-1&&a[i]>=a[j])
				ISAP.addedge(j,i,1);
	}
	ISAP.addedge(S,1,n);
	if(f[n]==ans)ISAP.addedge(n,T,n);
	printf("%d\n",ISAP.Max_Flow(S,T));
	return 0;
}