bzoj 3532
很好的一道题,对理解最小割有很大帮助
首先,不难发现本题与网络流24题中的某一道很类似,我们可以先跑一次dp求出每个节点的LIS,然后拆点,拆出的两点之间连流量为删除的代价的边,剩下的点之间按dp的转移连流量正无穷的边,最后跑最小割即为第一问答案
但是第二问有个问题:又引入了一个量要求最小割字典序最小,这怎么办?
首先我们考虑:字典序最小的话我们就要让第一个尽可能小,然后让第二个尽可能小...以此类推
那么我们考虑什么样的边可能在最小割中
这样的边一定满足如下条件:
第一.这条边已经满流了
第二.这条边的两端之间不存在增广路
如果一条边满足这样的条件,那么这条边就可以在一个最小割之中
那么我们把点按第三个变量排序,从小到大判断能否加入最小割,如果能加入最小割的话则需要退流,也就是退掉源点和汇点到这条边两端的流,这一步可以反向dfs解决
于是这题就结束了
#include <cstdio>
#include <cmath>
#include <cstring>
#include <cstdlib>
#include <iostream>
#include <algorithm>
#include <queue>
#include <stack>
using namespace std;
const int inf=0x3f3f3f3f;
struct Edge
{
int nxt;
int to;
int val;
int flo;
}edge[500005];
struct Info
{
int num,v,w,c;
friend bool operator < (Info a,Info b)
{
return a.c<b.c;
}
}f[1005];
int re[1005];
int head[5005];
int cur[5005];
int dis[5005];
int toe[5005];
int dp[1005];
int cnt=1,ttop=0;
int T,n;
int st,ed;
void init()
{
memset(head,0,sizeof(head));
memset(dp,0,sizeof(dp));
cnt=1,ttop=0;
}
void add(int l,int r,int w)
{
edge[cnt].nxt=head[l];
edge[cnt].to=r;
edge[cnt].val=w;
head[l]=cnt++;
}
void dadd(int l,int r,int w)
{
add(l,r,w),add(r,l,0);
}
int ide(int x)
{
return x&1?x+1:x-1;
}
bool bfs(int fr,int Ed)
{
memcpy(cur,head,sizeof(head));
memset(dis,0,sizeof(dis));
dis[fr]=1;
queue <int> M;
M.push(fr);
while(!M.empty())
{
int u=M.front();
M.pop();
for(int i=head[u];i;i=edge[i].nxt)
{
int to=edge[i].to;
if(edge[i].val>edge[i].flo&&!dis[to])
{
dis[to]=dis[u]+1,M.push(to);
if(to==Ed)return 1;
}
}
}
return dis[Ed];
}
int dfs(int x,int lim,int Ed)
{
if(x==Ed||!lim)return lim;
int ret=0;
for(int i=cur[x];i;i=edge[i].nxt)
{
cur[x]=i;
int to=edge[i].to;
if(dis[to]==dis[x]+1&&edge[i].val>edge[i].flo)
{
int temp=dfs(to,min(lim,edge[i].val-edge[i].flo),Ed);
if(temp)
{
ret+=temp,lim-=temp;
edge[i].flo+=temp,edge[ide(i)].flo-=temp;
if(!lim)break;
}
}
}
return ret;
}
int dinic()
{
int ans=0;
while(bfs(st,ed))ans+=dfs(st,inf,ed);
return ans;
}
inline int read()
{
int f=1,x=0;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
return x*f;
}
int main()
{
T=read();
while(T--)
{
init();
n=read();
st=2*n+1,ed=2*n+2;
for(int i=1;i<=n;i++)f[i].num=i,f[i].v=read(),dp[i]=1;
for(int i=1;i<=n;i++)f[i].w=read(),dadd((i<<1)-1,i<<1,f[i].w),toe[i]=cnt-1;
for(int i=1;i<=n;i++)for(int j=1;j<i;j++)if(f[i].v>f[j].v)dp[i]=max(dp[i],dp[j]+1);
int maxx=0;
for(int i=1;i<=n;i++)maxx=max(maxx,dp[i]);
for(int i=1;i<=n;i++)
{
if(dp[i]==1)dadd(st,(i<<1)-1,inf);
else if(dp[i]==maxx)dadd(i<<1,ed,inf);
for(int j=1;j<i;j++)if(f[i].v>f[j].v&&dp[j]+1==dp[i])dadd(j<<1,(i<<1)-1,inf);
}
for(int i=1;i<=n;i++)f[i].c=read();
printf("%d ",dinic());
sort(f+1,f+n+1);
for(int i=1;i<=n;i++)
{
int k=f[i].num;
if(bfs((k<<1)-1,k<<1))continue;
re[++ttop]=k;
while(bfs(ed,(k<<1)))dfs(ed,inf,(k<<1));
while(bfs((k<<1)-1,st))dfs((k<<1)-1,inf,st);
edge[toe[k]].val=edge[ide(toe[k])].val=0;
edge[toe[k]].flo=edge[ide(toe[k])].flo=0;
}
printf("%d\n",ttop);
sort(re+1,re+ttop+1);
for(int i=1;i<=ttop;i++)printf("%d ",re[i]);
printf("\n");
}
return 0;
}
bzoj 3532的更多相关文章
- BZOJ 3532: [Sdoi2014]Lis (最大流)
题目链接:http://www.lydsy.com:808/JudgeOnline/problem.php?id=3532 题意:给出三个数列ABC,长度均为n.删除A中的某些数字,使得A的最长上升子 ...
- BZOJ.3532.[SDOI2014]LIS(最小割ISAP 退流)
BZOJ 洛谷 \(LIS\)..经典模型? 令\(f_i\)表示以\(i\)结尾的\(LIS\)长度. 如果\(f_i=1\),连边\((S,i,INF)\):如果\(f_i=\max\limits ...
- [BZOJ]3532: [Sdoi2014]Lis
Time Limit: 10 Sec Memory Limit: 512 MB Description 给定序列A,序列中的每一项Ai有删除代价Bi和附加属性Ci.请删除若干项,使得4的最长上升子序 ...
- Lis(bzoj 3532)
Description 给定序列A,序列中的每一项Ai有删除代价Bi和附加属性Ci.请删除若干项,使得4的最长上升子序列长度减少至少1,且付出的代价之和最小,并输出方案. 如果有多种方案,请输出 ...
- BZOJ 2127: happiness [最小割]
2127: happiness Time Limit: 51 Sec Memory Limit: 259 MBSubmit: 1815 Solved: 878[Submit][Status][Di ...
- BZOJ 3275: Number
3275: Number Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 874 Solved: 371[Submit][Status][Discus ...
- BZOJ 2879: [Noi2012]美食节
2879: [Noi2012]美食节 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 1834 Solved: 969[Submit][Status] ...
- bzoj 4610 Ceiling Functi
bzoj 4610 Ceiling Functi Description bzoj上的描述有问题 给出\(n\)个长度为\(k\)的数列,将每个数列构成一个二叉搜索树,问有多少颗形态不同的树. Inp ...
- BZOJ 题目整理
bzoj 500题纪念 总结一发题目吧,挑几道题整理一下,(方便拖板子) 1039:每条线段与前一条线段之间的长度的比例和夹角不会因平移.旋转.放缩而改变,所以将每条轨迹改为比例和夹角的序列,复制一份 ...
- 【sdoi2013】森林 BZOJ 3123
Input 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整数N,M,T,分别表示节点数.初始边数.操作数.第三行包含N个非负整数 ...
随机推荐
- uniapp支付返回form
在项目中,支付功能是一个常见的功能,调用支付宝时,后段给我们的是一个form的富文本内容 这时需要把form到页面渲染,方法如下: uni.request({ //仅为示例,并非真实接口地址. url ...
- Delphi 自定义窗体类名
原理就是覆盖原CreateParams函数,重写新CreateParams函数,在新CreateParams函数继承完之后马上修改Parames.WinClassName type TForm1 = ...
- docker中搭建Ubuntu:16.04+python3.6+django环境
用vim 新建一个Dockerfile和sources.list文件,在里面添加如下内容. #sources.list deb http://mirrors.163.com/ubuntu/ xenia ...
- VMWARE 虚拟机配置优化
如果硬件性能不足,可做如下优化. 1.禁用 VMWARE 虚拟内存功能. 编辑->首选项-> 内存 , 设置如下,禁用内存交换. 2. 如果虚拟机装在机械盘,而电脑有固太硬盘,可通过 ...
- 使用虚拟环境-Python虚拟环境的安装和配置-virtualenv
打开windows命令终端(cmd)安装虚拟环境 virtualenv(如果你执行了上面查看python版本的语句,那么要先使用exit()方法先退出) pip3 install virtualenv ...
- 使用Visual Studio工具将ActiveX控件的COM类库转换为窗体控件
1:转换目的 Windows窗体只能继承Windows 窗体控件.如果要承载ActiveX控件,必须生成AxHost派生的包装器控件. 在使用WPF呈现ActiveX控件的时候,需要将com类库(oc ...
- SVD动态图
library(gganimate) library(gifski) # 构造数据 phi<-pi/6 theta<-pi/3 c<-matrix(c(cos(phi),sin(ph ...
- Day 22 22.2:scrapy部署
scrapy项目部署 scrapyd部署工具介绍 scrapyd是一个用于部署和运行scrapy爬虫的程序,它由 scrapy 官方提供的.它允许你通过JSON API来部署爬虫项目和控制爬虫运行. ...
- IntelliJ IDEA 生成serialVersionUID
1.设置Settings-->Editor->Inspections 2.实现Serializble
- iview table配合Tooltip使用的时候 气泡错位的问题
Tooltip 气泡错位 点击table的单元格,实现遮盖文字的预览功能,但是气泡会发生偏移,看了Tooltip的样式发现 气泡是绝对定位, 想到子绝父相,于是给表格的td添加相对定位的功能,结果能解 ...