【刷题】LOJ 6005 「网络流 24 题」最长递增子序列
题目描述
给定正整数序列 \(x_1 \sim x_n\) ,以下递增子序列均为非严格递增。
计算其最长递增子序列的长度 \(s\) 。
计算从给定的序列中最多可取出多少个长度为 \(s\) 的递增子序列。
如果允许在取出的序列中多次使用 \(x_1\) 和 \(x_n\) ,则从给定序列中最多可取出多少个长度为 \(s\) 的递增子序列。
输入格式
文件第 \(1\) 行有 \(1\) 个正整数 \(n\) ,表示给定序列的长度。接下来的 \(1\) 行有 \(n\) 个正整数 \(x_1 \sim x_n\) 。
输出格式
第 \(1\) 行是最长递增子序列的长度 \(s\) 。第 \(2\) 行是可取出的长度为 \(s\) 的递增子序列个数。第 \(3\) 行是允许在取出的序列中多次使用 \(x_1\) 和 \(x_n\) 时可取出的长度为 \(s\) 的递增子序列个数。
样例
样例输入
4
3 6 2 5
样例输出
2
2
3
数据范围与提示
\(1 \leq n \leq 500\)
题解
先用dp求出第一问的答案,和 \(f\) 数组,\(f[i]\) 代表以 \(i\) 为起点最长不下降子序列的长度
对于第二问,源点向 \(f[i]\) 等于第一问答案的点连边,\(f[i]=1\) 的点向汇点连边,中间的点 \(u\) 和点 \(v\) ,如果 \(f[u]=f[v]+1\) 且 \(a[u] \leq a[v]\) ,那么它们之间连边,这些边容量均为 \(1\) 。并且因为每个点只能用一次,所以拆点,中间连容量为 \(1\) 的边。跑最大流就是第二问的答案
对于第三问,把一号点和最后一个点的容量设为 \(inf\) 就好了
#include<bits/stdc++.h>
#define ui unsigned int
#define ll long long
#define db double
#define ld long double
#define ull unsigned long long
const int MAXN=1000+10,MAXM=MAXN*MAXN+10,inf=0x3f3f3f3f;
int n,a[MAXN],f[MAXN],e=1,beg[MAXN],cur[MAXN],level[MAXN],nex[MAXM<<1],to[MAXM<<1],cap[MAXM<<1],clk,vis[MAXN],s,t,ans1;
std::queue<int> q;
template<typename T> inline void read(T &x)
{
T data=0,w=1;
char ch=0;
while(ch!='-'&&(ch<'0'||ch>'9'))ch=getchar();
if(ch=='-')w=-1,ch=getchar();
while(ch>='0'&&ch<='9')data=((T)data<<3)+((T)data<<1)+(ch^'0'),ch=getchar();
x=data*w;
}
template<typename T> inline void write(T x,char ch='\0')
{
if(x<0)putchar('-'),x=-x;
if(x>9)write(x/10);
putchar(x%10+'0');
if(ch!='\0')putchar(ch);
}
template<typename T> inline void chkmin(T &x,T y){x=(y<x?y:x);}
template<typename T> inline void chkmax(T &x,T y){x=(y>x?y:x);}
template<typename T> inline T min(T x,T y){return x<y?x:y;}
template<typename T> inline T max(T x,T y){return x>y?x:y;}
inline void insert(int x,int y,int z)
{
to[++e]=y;
nex[e]=beg[x];
beg[x]=e;
cap[e]=z;
to[++e]=x;
nex[e]=beg[y];
beg[y]=e;
cap[e]=0;
}
inline void dp()
{
for(register int i=1;i<=n;++i)f[i]=1;
for(register int i=n-1;i>=1;--i)
for(register int j=n;j>i;--j)
if(a[j]>=a[i])chkmax(f[i],f[j]+1);
for(register int i=1;i<=n;++i)chkmax(ans1,f[i]);
}
inline bool bfs()
{
memset(level,0,sizeof(level));
level[s]=1;
q.push(s);
while(!q.empty())
{
int x=q.front();
q.pop();
for(register int i=beg[x];i;i=nex[i])
if(cap[i]&&!level[to[i]])level[to[i]]=level[x]+1,q.push(to[i]);
}
return level[t];
}
inline int dfs(int x,int maxflow)
{
if(x==t||!maxflow)return maxflow;
int res=0;
vis[x]=clk;
for(register int &i=cur[x];i;i=nex[i])
if((vis[to[i]]^vis[x])&&cap[i]&&level[to[i]]==level[x]+1)
{
int nf=dfs(to[i],min(maxflow,cap[i]));
res+=nf;
cap[i]-=nf;
cap[i^1]+=nf;
maxflow-=nf;
if(!maxflow)break;
}
vis[x]=0;
return res;
}
inline int Dinic()
{
int res=0;
while(bfs())clk++,memcpy(cur,beg,sizeof(cur)),res+=dfs(s,inf);
return res;
}
int main()
{
read(n);
for(register int i=1;i<=n;++i)read(a[i]);
dp();
s=n+n+1,t=s+1;
write(ans1,'\n');
for(register int i=1;i<=n;++i)
{
insert(i,i+n,1);
if(f[i]==ans1)insert(s,i,1);
if(f[i]==1)insert(i+n,t,1);
for(register int j=i+1;j<=n;++j)
if(a[j]>=a[i]&&f[i]==f[j]+1)insert(i+n,j,1);
}
write(Dinic(),'\n');
if(ans1==1)write(n,'\n');
else
{
e=0;clk=0;
memset(beg,0,sizeof(beg));
for(register int i=1;i<=n;++i)
{
if(i==1||i==n)
{
insert(i,i+n,inf);
if(f[i]==ans1)insert(s,i,inf);
if(f[i]==1)insert(i+n,t,inf);
}
else
{
insert(i,i+n,1);
if(f[i]==ans1)insert(s,i,1);
if(f[i]==1)insert(i+n,t,1);
}
for(register int j=i+1;j<=n;++j)
if(a[j]>=a[i]&&f[i]==f[j]+1)insert(i+n,j,1);
}
write(Dinic(),'\n');
}
return 0;
}
【刷题】LOJ 6005 「网络流 24 题」最长递增子序列的更多相关文章
- 【刷题】LOJ 6227 「网络流 24 题」最长k可重线段集问题
题目描述 给定平面 \(\text{xoy}\) 上 \(n\) 个开线段组成的集合 \(\text{I}\) ,和一个正整数 \(k\) ,试设计一个算法. 从开线段集合 \(\text{I}\) ...
- Libre 6005 「网络流 24 题」最长递增子序列 / Luogu 2766 最长递增子序列问题(网络流,最大流)
Libre 6005 「网络流 24 题」最长递增子序列 / Luogu 2766 最长递增子序列问题(网络流,最大流) Description 问题描述: 给定正整数序列x1,...,xn . (1 ...
- [luogu_P1251][LOJ#6008]「网络流 24 题」餐巾计划
[luogu_P1251][LOJ#6008]「网络流 24 题」餐巾计划 试题描述 一个餐厅在相继的 \(N\) 天里,第 \(i\) 天需要 \(R_i\) 块餐巾 \((i=l,2,-,N)\) ...
- [LOJ#6002]「网络流 24 题」最小路径覆盖
[LOJ#6002]「网络流 24 题」最小路径覆盖 试题描述 给定有向图 G=(V,E).设 P 是 G 的一个简单路(顶点不相交)的集合.如果 V 中每个顶点恰好在 P 的一条路上,则称 P 是 ...
- loj #6014. 「网络流 24 题」最长 k 可重区间集
#6014. 「网络流 24 题」最长 k 可重区间集 题目描述 给定实直线 L LL 上 n nn 个开区间组成的集合 I II,和一个正整数 k kk,试设计一个算法,从开区间集合 I II 中选 ...
- loj #6013. 「网络流 24 题」负载平衡
#6013. 「网络流 24 题」负载平衡 题目描述 G 公司有 n nn 个沿铁路运输线环形排列的仓库,每个仓库存储的货物数量不等.如何用最少搬运量可以使 n nn 个仓库的库存数量相同.搬运货物时 ...
- loj #6122. 「网络流 24 题」航空路线问题
#6122. 「网络流 24 题」航空路线问题 题目描述 给定一张航空图,图中顶点代表城市,边代表两个城市间的直通航线.现要求找出一条满足下述限制条件的且途经城市最多的旅行路线. 从最西端城市出发,单 ...
- loj #6121. 「网络流 24 题」孤岛营救问题
#6121. 「网络流 24 题」孤岛营救问题 题目描述 1944 年,特种兵麦克接到国防部的命令,要求立即赶赴太平洋上的一个孤岛,营救被敌军俘虏的大兵瑞恩.瑞恩被关押在一个迷宫里,迷宫地形复杂, ...
- loj #6226. 「网络流 24 题」骑士共存问题
#6226. 「网络流 24 题」骑士共存问题 题目描述 在一个 n×n\text{n} \times \text{n}n×n 个方格的国际象棋棋盘上,马(骑士)可以攻击的棋盘方格如图所示.棋盘上 ...
随机推荐
- 关于PCB的线宽与过孔
关于PCB的线宽与过孔 我们在画PCB时一般都有一个常识,即走大电流的地方用粗线(比如50mil,甚至以上),小电流的信号可以用细线(比如10mil). 对于某些机电控制系统来说,有时候走线里流过的瞬 ...
- C# event线程安全
突然想到有关C#中使用event特性时关于线程安全的问题,以前虽然有遵从“复制引用+null判断”的模式(盲目地),但没有深入了解和思考. 为之查询了资料和实验,对此有了进一步的理解. 一般event ...
- [CF1038F]Wrap Around[AC自动机+dp]
题意 题目链接 分析 题意容易转化成求循环之后不包含 \(s\) 的串的个数. 首先建立 AC 自动机.考虑一个暴力的做法:枚举长度为 \(n\) 的字符串 \(t\) 最终(后缀) 和 \(s\) ...
- Macaca 基础原理浅析
导语 前面几篇文章介绍了在Macaca实践中的一些实用技巧与解决方案,今天简单分析一下Macaca的基础原理.这篇文章将以前面所分享的UI自动化Macaca-Java版实践心得中的demo为基础,进行 ...
- 利用fiddler core api 拦截修改 websocket 数据
一般的中间人攻击基本都是拦截修改普通的http协议里面的内容,而对于怎么拦截修改websocket协议传输的内容好像都没有多少介绍. talk is cheap show me the code us ...
- 用Spring.Services整合 thrift0.9.2生成的wcf中间代码-复杂的架构带来简单的代码和高可维护性
最近一直在看关于thrift的相关文章,涉及到的内容的基本都是表层的.一旦具体要用到实际的项目中的时候就会遇到各种问题了! 比如说:thrift 的服务器端载体的选择.中间代码的生成options(a ...
- 关于CMS的那点事 I
CMS是Content Management System的缩写,意为"内容管理系统".CMS其实是一个很广泛的称呼,从一般的博客程序,新闻发布程序,到综合性的网站管理程序都可以被 ...
- EF Core 新特性——Owned Entity Types
Owned Entity Types 首先owned entity type是EF Core 2.0的新特性. 至于什么是owned entity types,可以先把他理解为EF Core官方支持的 ...
- 关于Map迭代循环,key和value的顺序问题
使用Hashtable,keySet()返回的顺序为降序(key降顺序) ---->6, 5, 4, 3, 2, 1使用TreeMap,keySet()返回的顺序为升序(key升顺序) ---- ...
- 《杜增强讲Unity之Tanks坦克大战》11-游戏流程控制
11 游戏流程控制 使用协程来控制游戏流程 11.1 添加MessageText 首先添加一个Text来显示文字 image 设置GameMgr image 11.2 游戏整体流程 下面Gam ...