ha~

«问题描述:

给定正整数序列$x_1,...,x_n$ 。$n<=500$

求(1)计算其最长不下降子序列的长度$s$。

(2)计算从给定的序列中最多可取出多少个长度为$s$的不下降子序列。

(3)如果允许在取出的序列中多次使用$x_1$和$x_n$,则从给定序列中最多可取出多少个长度为$s$的不下降子序列。


(1)暴力n方解决

(2)建分层图,把图每个顶点i按照F[i]的不同分为若干层,这样图中从S出发到T的任何一条路径都是一个满足条件的最长不下降子序列。由 S 向所有$ f_i = 1$ 的$ i $连容量为$ 1 $的边,由 所有$ f_i = s $的$ i $向$ T $连容量为$ 1$ 的边。对于所有的$ (i,j)$,若$ i < j,f_i +1 = f_j$,则由$ i$ 向$ j$ 连容量为 $1$ 的边。这样一来,所 有从 $S$ 出发到达$ T$ 的的路径都是一个长度为 $s $的上升子序 列。

注意拆点,以保证同一个数不会重复使用。一个连入边,一个连出边,两个点 之间连流量为 1 的限制边。

注意事项: line49的条件判断注意$f[j]+1==f[i]$的前提是要$a[j]<=a[i]$才可以连边!

(3)把之前的边的流量限制改一下即可,对1号和n号微调后原样跑。

注意事项(其实网上绝大部分题解好像都忽略了):建边时s向1号点的容量不可为无穷,应设为n,  n号点连t同理.

比如这个毒瘤数据

1

1

绝大部分题解输的是INF,但其实是1

 #include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<int,int> pii;
template<typename T>inline char MIN(T&A,T B){return A>B?A=B,:;}
template<typename T>inline char MAX(T&A,T B){return A<B?A=B,:;}
template<typename T>inline T _min(T A,T B){return A<B?A:B;}
template<typename T>inline T _max(T A,T B){return A>B?A:B;}
template<typename T>inline T read(T&x){
x=;char c;while(!isdigit(c=getchar()))if(isalpha(c))return x=(int)c;
while(isdigit(c))x=(x<<)+(x<<)+(c^),c=getchar();return x;
}
const int N=+,M=+,INF=0x3f3f3f3f;
int w[][M<<],v[][M<<],Next[][M<<],Head[][N<<],cur[][N<<],dis[][N<<],tot[],s,t,n;
inline void Addedge(int x,int y,int z,int p){
v[p][++tot[p]]=y,Next[p][tot[p]]=Head[p][x],Head[p][x]=tot[p],w[p][tot[p]]=z;
v[p][++tot[p]]=x,Next[p][tot[p]]=Head[p][y],Head[p][y]=tot[p],w[p][tot[p]]=;
}
#define y v[p][j]
inline char bfs(int p){
queue<int> q;q.push(s),memset(dis[p],,sizeof dis[p]),dis[p][s]=;
for(register int i=;i<=(n<<)+;++i)cur[p][i]=Head[p][i];
while(!q.empty()){
int x=q.front();q.pop();
for(register int j=Head[p][x];j;j=Next[p][j])if(w[p][j]&&!dis[p][y]){
dis[p][y]=dis[p][x]+,q.push(y);
if(y==t)return ;
}
}
return ;
}
int dinic(int x,int flow,int p){
if(!flow||x==t)return flow;
int rest=flow,k;
for(register int j=cur[p][x];j&&rest;cur[p][x]=j,j=Next[p][j])if(w[p][j]&&dis[p][y]==dis[p][x]+){
if(!(k=dinic(y,_min(rest,w[p][j]),p)))dis[p][y]=;
rest-=k,w[p][j]-=k,w[p][j^]+=k;
}
return flow-rest;
}
#undef y
int a[N],f[N],ans,maxflow; int main(){//freopen("P2766.in","r",stdin);//freopen("P2766.txt","w",stdout);
read(n);s=(n<<)+,t=s+,tot[]=tot[]=;
for(register int i=;i<=n;++i){
read(a[i]);f[i]=;
for(register int j=;j<i;++j)if(a[j]<=a[i])MAX(f[i],f[j]+);
for(register int j=;j<i;++j)if(f[j]+==f[i]&&a[j]<=a[i])Addedge(j+n,i,,),Addedge(j+n,i,,);
MAX(ans,f[i]),Addedge(i,i+n,,);if(f[i]==)Addedge(s,i,,),i==?Addedge(s,i,n,):Addedge(s,i,,);
if(i==||i==n)Addedge(i,i+n,n,);else Addedge(i,i+n,,);
}
printf("%d\n",ans);
for(register int i=;i<=n;++i)if(f[i]==ans)Addedge(i+n,t,,),i==n?Addedge(i+n,t,n,):Addedge(i+n,t,,);
while(bfs())maxflow+=dinic(s,n,);
printf("%d\n",maxflow);maxflow=;
while(bfs())maxflow+=dinic(s,n,);
printf("%d\n",maxflow);
return ;
}

最新文章

  1. (整理)MyBatis入门教程(一)
  2. 剖析AngularJS作用域
  3. TortoiseGit 连接Git服务器不用每次输入用户名和密码的方法
  4. JAVA 回调机制(callback)
  5. Python操作PDF与Tiff文件
  6. User Agent跨站攻击
  7. 2016百度之星-初赛(Astar Round2A)AII X
  8. 由csdn开源项目评选中闹出刷票问题想到投票程序的设计
  9. wildfly-9.0.2 web项目部署详细步骤
  10. 用Jpush极光推送实现抓取特定某个用户Log到七牛云服务器
  11. Js与Jq 获取浏览器和对象值的方法
  12. hdu 5311 Hidden String(find,substr)
  13. 使用反射+策略模式代替项目中大量的switch case判断
  14. Python基础学习参考(三):内置函数
  15. python--装饰器(附偏函数、断言)
  16. 生产环境ssh登陆策略
  17. 线性求第k大
  18. XBee&#174; ZigBee 模块使用方法
  19. PLSQL基础学习-文字
  20. django403错误(转)

热门文章

  1. linux 源代码安装mysql5.5
  2. 优秀JS学习站点
  3. Android-Animations介绍
  4. IOS-4-面试题1:黑马程序猿IOS面试题大全
  5. 目标检测之人头检测(HaarLike Adaboost)---高密度环境下行人检测和统计
  6. DB2 时间操作
  7. share初始化
  8. vue-cli3.0升级失败,vue-cli卸载不掉,vue-cli升级不了3.0
  9. 实现RTSP摄像机进行网页直播和微信直播的技术方案:EasyNVR自动更新方法
  10. EasyNVR无插件IPC摄像机直播方案前端构建之:如何区分PC端和移动端