「luogu2766」最长不下降子序列问题

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 const int N=510,M=251000,oo=2e9;
 4 int n,a[N],f[N],ss,tt;
 5 struct Edge{
 6     int from,to,flow,cap;
 7     Edge(int _from=0,int _to=0,int _flow=0,int _cap=0):from(_from),to(_to),flow(_flow),cap(_cap){}
 8 };
 9 Edge edge[M<<1];
10 int edge_tot,spe[10];
11 vector<int>point[N<<1];
12 void add_edge(int f,int t,int c){
13     edge[edge_tot]=Edge(f,t,0,c);
14     point[f].push_back(edge_tot++);
15     edge[edge_tot]=Edge(t,f,0,0);
16     point[t].push_back(edge_tot++);
17     return;
18 }
19 int level[N<<1];
20 bool bfs(){
21     memset(level,0,sizeof(level));
22     queue<int>q;
23     q.push(ss);
24     level[ss]=1;
25     int x;
26     while(!q.empty()){
27         x=q.front();q.pop();
28         for(int i=0;i<point[x].size();i++){
29             Edge& e=edge[point[x][i]];
30             if(e.cap>e.flow&&!level[e.to]){
31                 level[e.to]=level[x]+1;
32                 q.push(e.to);
33             }
34         }
35     }
36     return level[tt];
37 }
38 int dfs(int k,int aa){
39     if(k==tt||!aa) return aa;
40     int temp,ans=0;
41     for(int i=0;i<point[k].size();i++){
42         Edge& e=edge[point[k][i]];
43         if(e.cap>e.flow&&level[e.to]==level[k]+1&&(temp=dfs(e.to,min(aa,e.cap-e.flow)))){
44             ans+=temp,aa-=temp;
45             e.flow+=temp,edge[point[k][i]^1].flow-=temp;
46         }
47         if(!aa) return ans;
48     }
49     return ans;
50 }
51 int dinic(){
52     int ans=0;
53     while(bfs()) ans+=dfs(ss,oo);
54     return ans;
55 }
56 int main(){
57 //  freopen("in.txt","r",stdin);
58 //  freopen("out.txt","w",stdout);
59     int ans1=0,ans2=0,ans3=0;
60     scanf("%d",&n);
61     ss=(n<<1)+1,tt=ss+1;
62     for(int i=1;i<=n;i++) scanf("%d",&a[i]);
63     a[n+1]=oo;
64     for(int i=n;i;i--){
65         for(int j=i+1;j<=n+1;j++) if(a[i]<=a[j]) f[i]=max(f[j]+1,f[i]);
66         ans1=max(ans1,f[i]);
67     }
68     printf("%d\n",ans1);
69     for(int i=1;i<=n;i++){
70         if(i==1||i==n) spe[++spe[0]]=edge_tot;
71         add_edge(i,i+n,1);
72         if(f[i]==ans1){
73             if(i==1) spe[++spe[0]]=edge_tot;
74             add_edge(ss,i,1);
75         }
76         if(f[i]==1){
77             if(i==n) spe[++spe[0]]=edge_tot;
78             add_edge(i+n,tt,1);
79         }
80     }
81     for(int i=1;i<=n;i++)
82         for(int j=i+1;j<=n;j++) if(a[j]>=a[i]&&f[j]==f[i]-1)
83             add_edge(i+n,j,1);
84     ans2=dinic();
85     printf("%d\n",ans2);
86     for(int i=0;i<edge_tot;i++) edge[i].flow=0;
87     for(int i=1;i<=spe[0];i++) edge[spe[i]].cap=oo;
88     ans3=dinic();
89     printf("%d\n",ans3);
90     return 0;
91 }

 

posted @ 2018-03-08 13:15  Cupcake  阅读(115)  评论(0编辑  收藏  举报