HDU 3998 Sequence (最长上升子序列+最大流)
参考链接:
http://www.cnblogs.com/gentleh/archive/2013/03/30/2989958.html
题意:求一个序列的最长上升子序列,及其个数(注意:两个最长上升子序列不能共有同一个数,才能算两个)
思路:用dp+最大流来求,首先用两次循环dp求出最长上升子序列的长度ans,然后就是建图了,可以选择源点为0,
由于数列中的每一个数只能使用一次,构图的时候需要拆点。若有n个数,则拆成2 * n个点,构造源点s=0和汇点t=2*n+1,
将每个点(i)都和自己拆出来的点(i+n)连边,将源点和dp[i]=1的点连边,将dp[i]=ans的点与汇点连边,
最后若dp[j] = dp[i] + 1,则将i + n和 j连边。所有边的流量都是1,这样便可以限制每个点只使用一次。
其实连边的顺序就是最长序列为1,2,3,...,ans。可以理解为从最长序列为1(该数本身)一直流到数列中的最长上升序列。
这里要注意的是,由于数据弱,有不用拆点也能A的。但是为了保证每个点都是用一次,一定要拆点。
如果不拆点的话,可以见下图,则中间那个点可能会被用两次,具体不仔细说了。

我一开始按照dinic模板打了一遍,结果TLE。。。后来经别人指点,加了dinic中的两个判断条件,见(1)和(2),瞬间A了。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include <queue>
#define INF 0x3f3f3f3f using namespace std;
const int maxn=;
int n,m;
int a[maxn]; //序列
int pri[maxn*];
int head[maxn*]; //因为要拆点
int s,t; //s:源点 t:汇点
int dp[maxn];
int tot=;
int ans,num;//最长上升子序列的长度,以及不同的个数。 struct Edge{
int from,to;
int next;
int c,f;
}edge[maxn*]; void add(int x,int y,int val){
edge[tot].next=head[x];
edge[tot].from=x;
edge[tot].to=y;
edge[tot].c=val; //注意这里是c为val,反向边c为0
//edge[tot].f=0;
head[x]=tot++; edge[tot].next=head[y];
edge[tot].from=y;
edge[tot].to=x;
edge[tot].c=;
//edge[tot].f=0;
head[y]=tot++;
}
bool BFS(){
queue<int>q;
memset(pri,-,sizeof(pri));
pri[]=;//源点
q.push();
while(!q.empty()){
int tmp=q.front();
q.pop();
for(int k=head[tmp];k!=-;k=edge[k].next){
int v=edge[k].to;
if(pri[v]==- && edge[k].c>){
pri[v]=pri[tmp]+;
if(v==t){
return true;
}
q.push(v);
}
}
}
return false;
} int dinic(int p,int flow){
if(p==t){
return flow;
}
int f=flow;
for(int k=head[p];k!=-;k=edge[k].next){
int v=edge[k].to;
if(pri[v]==pri[p]+ && edge[k].c>){
int a=edge[k].c; //a为该边可以增加的流量
int ff=dinic(v,min(a,flow)); //ff为路径中所有a的最小值,即为该条路中可以增加的流量
//edge[k].f+=ff;
//edge[k^1].f-=ff; //因为双向边是同时建的,编号必定一奇一偶成对的,如0、1;2、3;4、5;
//这样用k异或1即可求得对应的反向边
edge[k].c-=ff;
edge[k^].c+=ff;
flow-=ff;
if(flow<=)
break; //(1)
}
}
if(f-flow<=)
pri[p]=-; //(2)
//如果从p点流出去的流量<=0,那么设置pri[p]的值为-1,之后在dinic中就不考虑到p点的情况了。
return f-flow;
} int main()
{
while(scanf("%d",&n)!=EOF){
for(int i=;i<=n;i++){
dp[i]=;
scanf("%d",&a[i]);
}
for(int i=;i<=n;i++){
for(int j=;j<i;j++){
if(a[i]>a[j] && dp[j]+>dp[i])
dp[i]=dp[j]+;
}
}
m=*n+;
ans=;
for(int i=;i<=n;i++){
ans=max(dp[i],ans);
}
memset(head,-,sizeof(head));
tot=; s=;
t=*n+;
for(int i=;i<=n;i++){
add(i,i+n,);
if(dp[i]==)
add(s,i,);
if(dp[i]==ans)
add(i+n,t,);
}
for(int i=;i<=n;i++){
//下面j只要从i+1开始即可,因为add加边是加双向边的
for(int j=i+;j<=n;j++){
if(dp[j]==dp[i]+){
add(i+n,j,);
}
}
}
num=;
while(BFS()){
num+=dinic(s,INF);
}
printf("%d\n%d\n",ans,num); }
return ;
}
HDU 3998 Sequence (最长上升子序列+最大流)的更多相关文章
- HDU 3998 Sequence(经典问题,最长上升子序列)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3998 解题报告:求一个数列的最长上升子序列,并求像这样长的不相交的子序列最多有多少个. 我用的是最简单 ...
- hdu 5748(求解最长上升子序列的两种O(nlogn)姿势)
Bellovin Time Limit: / MS (Java/Others) Memory Limit: / K (Java/Others) Total Submission(s): Accepte ...
- HDU 4681 String 最长公共子序列
题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=4681 题意: 给你a,b,c三个串,构造一个d串使得d是a,b的子序列,并且c是d的连续子串.求d最大 ...
- HDU 1159.Common Subsequence-最长公共子序列(LCS)
Common Subsequence Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Other ...
- hdu 3998 Sequence
There is a sequence X (i.e. x[1], x[2], ..., x[n]). We define increasing subsequence of X as x[i1], ...
- hdu 1025 dp 最长上升子序列
//Accepted 4372 KB 140 ms //dp 最长上升子序列 nlogn #include <cstdio> #include <cstring> #inclu ...
- hdu 5489(LIS最长上升子序列)
题意:一个含有n个元素的数组,删去k个连续数后,最长上升子序列 /*思路参考GoZy 思路: 4 2 3 [5 7 8] 9 11 ,括号表示要删掉的数, 所以 最长上升子序列 = ...
- hdu 5532(最长上升子序列)
Input The first line contains an integer T indicating the total number of test cases. Each test case ...
- Cogs 731. [网络流24题] 最长递增子序列(最大流)
[网络流24题] 最长递增子序列 ★★★☆ 输入文件:alis.in 输出文件:alis.out 简单对比 时间限制:1 s 内存限制:128 MB «问题描述: 给定正整数序列x1,-, xn. ( ...
随机推荐
- C语言 将产生的随机数存入数组,数据不能相同
1.定义一个一维数,数组大小为24. 2.产生0~23的随机数. 3.将产生的随机数存入i数组,要求数组中的每个数据不能相同. 4.补充说明,这个子程序要求每次调用后,这个数组里面就 存放了0~23这 ...
- ED/EP系列1《简介》
电子存折(ED:ElectronicDeposit)一种为持卡人进行消费.取现等交易而设计的支持个人识别码(PIN)保护的金融IC卡应用.它支持圈存.圈提.消费和取现等交易. 电子钱包(EP:Elec ...
- Redbean:入门(四) - 反射机制 以及 事务
<?php //引入rb入口文件 include_once 'rb.php'; //定义dsn以及相关的数据 $dsn = 'mysql:host=localhost;dbname=hwibs_ ...
- Android--启动拍照功能并返回结果
因为没有深入学习拍照这块功能,所以只是简单的调用了一下系统的拍照功能,下面代码: //拍照的方法 private void openTakePhoto(){ /** * 在启动拍照之前最好先判断一下s ...
- VS中引用第三方库的方法(配置sqlite数据库)
我们在编写程序时,,不可避免的会使用第三方的库文件,很少使用源文件(.cpp),大部分是使用对类进行声明的头文件和封装了类的链接库(静态lib或动态dll),比如我们写程序用的iostream这个库, ...
- angular2 国际化实现
angular2国际化通过管道(pipe)的形式实现下载ng2-translate 如何使用可以参照https://github.com/ocombe/ng2-translate 自己写了一个小DEM ...
- 简单的C语言小学四则运算设计
题目:设计一个简单的四则运算编辑器 思路:我使用的是C语言编程,看到题目首先要随机出3个随机数,其中两个为100以内的随机数(a,b),一个为0~3的随机数(k). k值的变化使得+ - * /的变化 ...
- SQL中的5种聚集函数
作为一个刚毕业进入这行的菜鸟,婶婶的觉的那种大神.大牛到底是怎样炼成的啊,我这小菜鸟感觉这TMD要学的东西这多啊,然后就给自己定了许多许多要学习的东西,可是有人又不停地给你灌输:东西不在多而要精通!我 ...
- java 24点算法实现
最近闲来无事,突然怀念起小时候和堂兄表姐们经常玩24点游戏,于是就琢磨着是不是开发一个安卓手机版本.然后上网上一搜,发现已经被别人给开发烂了啊.不过这只能说明这个小游戏要想赚广告费很难了,但是拿来锻炼 ...
- Windows下查看8080进程及结束进程命令
Windows下查看进程及结束进程命令 1)查看占用8080端口的进程号 >netstat –aon | findstr “8080” 结果:TCP 0.0.0.0:8080 ...