[Wf2011]Chips Challenge
两个条件都不太好处理
每行放置的个数实际很小,枚举最多放x
但还是不好放
考虑所有位置先都放上,然后删除最少使得合法
为了凑所有的位置都考虑到,把它当最大流
但是删除最少,所以最小费用
行列相关,左行点,右列点
S到行,流“能填位置”费0
列到T,流“能填位置”费0
i行到i列,流x,即枚举的最大个数
空位(i,j),i行连j列,流1费0
最小费用最大流
意义:流过i行到i列的流量,象征留下一个芯片
流过费用为1的,象征把这个芯片删除。
最大流保证所有位置都考虑到了
最小费用使得最少。
可以发现最后的结果一定满足条件1
条件2?
最大流为flow,费用为cos,总共的位置(多出来的+必填)=sum
放置了tot=sum-cos
如果有x/tot<=A/B那么更新ans=max(ans,tot)
x一定时,tot越大,越可能比A/B小。和最小费用相符。
虽然可能x过大,但是答案一定可以枚举到。
代码:
#include<bits/stdc++.h>
#define reg register int
#define il inline
#define numb (ch^'0')
using namespace std;
typedef long long ll;
il void rd(int &x){
char ch;x=;bool fl=false;
while(!isdigit(ch=getchar()))(ch=='-')&&(fl=true);
for(x=numb;isdigit(ch=getchar());x=x*+numb);
(fl==true)&&(x=-x);
}
namespace Miracle{
const int N=;
const int P=;
const int inf=0x3f3f3f3f;
int n,A,B;
struct node{
int nxt,to;
int c,w;
}e[*(N*N+*N)+];
int hd[P],cnt=;
int s,t;
void add(int x,int y,int w,int c){
e[++cnt].nxt=hd[x];
e[cnt].to=y;
e[cnt].c=c;
e[cnt].w=w;hd[x]=cnt; e[++cnt].nxt=hd[y];
e[cnt].to=x;
e[cnt].c=-c;
e[cnt].w=;hd[y]=cnt;
}
int dis[P];
bool in[P];
int incf[P],pre[P];
queue<int>q;
int ans,flow,cos;
int l[N],h[N];
char mp[N][N];
bool spfa(){
memset(dis,inf,sizeof dis);
memset(pre,,sizeof pre);
while(!q.empty()) q.pop();
dis[s]=;
incf[s]=inf;
q.push(s);
while(!q.empty()){
int x=q.front();q.pop();
in[x]=;
for(reg i=hd[x];i;i=e[i].nxt){
int y=e[i].to;
if(e[i].w){
if(dis[y]>dis[x]+e[i].c){
dis[y]=dis[x]+e[i].c;
pre[y]=i;
incf[y]=min(incf[x],e[i].w);
if(!in[y]){
in[y]=;
q.push(y);
}
}
}
}
}
if(dis[t]==inf) return false;
return true;
}
void upda(){
int x=t;
while(pre[x]){
e[pre[x]].w-=incf[t];
e[pre[x]^].w+=incf[t];
x=e[pre[x]^].to;
}
flow+=incf[t];
cos+=dis[t]*incf[t];
}
void EK(int lim){
cos=;flow=;
memset(hd,,sizeof hd);
cnt=;
s=,t=*n+;
for(reg i=;i<=n;++i) {
add(s,i,l[i],);
add(i+n,t,h[i],);
add(i,i+n,lim,);
}
for(reg i=;i<=n;++i){
for(reg j=;j<=n;++j){
if(mp[i][j]=='.') add(i,j+n,,);
}
}
while(spfa()) upda();
}
void clear(){
ans=-;
for(reg i=;i<=n;++i) l[i]=n;
for(reg j=;j<=n;++j) h[j]=n;
}
int main(){
int o=;
while(){
rd(n);rd(A);rd(B);
if(n==&&A==&&B==) break;
++o;
clear();
int can=;
int alr=;
for(reg i=;i<=n;++i){
scanf("%s",mp[i]+);
for(reg j=;j<=n;++j){
if(mp[i][j]=='/') --l[i],--h[j];
else ++can;
if(mp[i][j]=='C') ++alr;
}
}
for(reg x=;x<=n;++x){
EK(x);
int tot=can-cos;
if(flow==can&&A*tot>=x*B) ans=max(ans,tot);
}
ans-=alr;
printf("Case %d: ",o);
if(ans<) printf("impossible");
else printf("%d",ans);
puts("");
}
return ;
} }
signed main(){
Miracle::main();
return ;
} /*
Author: *Miracle*
Date: 2019/1/8 10:23:22
*/
最小费用最大流可以考虑两个限制
最大流限制合法
费用流限制优秀
[Wf2011]Chips Challenge的更多相关文章
- 【BZOJ 2673】[Wf2011]Chips Challenge
题目大意: 传送门 $n*n$的棋盘,有一些位置可以放棋子,有一些已经放了棋子,有一些什么都没有,也不能放,要求放置以后满足:第i行和第i列的棋子数相同,同时每行的棋子数占总数比例小于$\frac{A ...
- Bzoj2673 3961: [WF2011]Chips Challenge 费用流
国际惯例题面:如果我们枚举放几个零件的话,第二个限制很容易解决,但是第一个怎么办?(好的,这么建图不可做)考虑我们枚举每行每列最多放几个零件t,然后计算零件总数sum.这样如果可行的话,则有t*B&l ...
- BZOJ2673 [Wf2011]Chips Challenge 费用流 zkw费用流 网络流
https://darkbzoj.cf/problem/2673 有一个芯片,芯片上有N*N(1≤N≤40)个插槽,可以在里面装零件. 有些插槽不能装零件,有些插槽必须装零件,剩下的插槽随意. 要求装 ...
- bzoj3961[WF2011]Chips Challenge
题意 给出一个n*n的网格,有些格子必须染成黑色,有些格子必须染成白色,其他格子可以染成黑色或者白色.要求最后第i行的黑格子数目等于第i列的黑格子数目,且某一行/列的格子数目不能超过格子总数的A/B. ...
- bzoj 3961: [WF2011]Chips Challenge【最小费用最大流】
参考:https://blog.csdn.net/Quack_quack/article/details/50554032 神建图系列 首先把问题转为全填上,最少扣下来几个能符合条件 先考虑第2个条件 ...
- [2011WorldFinal]Chips Challenge[流量平衡]
Chips Challenge Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others) ...
- 【UVALive - 5131】Chips Challenge(上下界循环费用流)
Description A prominent microprocessor company has enlisted your help to lay out some interchangeabl ...
- 解题:BZOJ 2673 World Final 2011 Chips Challenge
题面 数据范围看起来很像网络流诶(滚那 因为限制多而且强,数据范围也不大,我们考虑不直接求答案,而是转化为判定问题 可以发现第二个限制相对好满足,我们直接枚举这个限制就可以.具体来说是枚举所有行中的最 ...
- 【题解】uva1104 chips challenge
原题传送门 题目分析 给定一张n*n的芯片. '.'表示该格子可以放一个零件. 'C'表示该格子已经放了一个零件(不能拆下). '/'表示该格子不能放零件. 要求在芯片的现有基础上,放置尽可能多的零件 ...
随机推荐
- Tensorflow基本开发架构
Tensorflow基本开发架构 先说句题外话, 这段时间一直研究爬虫技术,主要目的是为将来爬取训练数据做准备,同时学习python编程.这一研究才发现,python的开发资源实在是太丰富了,所有你能 ...
- python-redis哈希模式
命令: hset info name hgetall info hkeys info hvlls info m系列批量处理: ---------------------------------- ...
- vs2017搭建linux c++开发环境
最近一直在阅读ovs的源码,看到用户态代码的时候,需要对用户态的代码进行调试,一开始想直接使用linux中的GDB进行调试,但是ovs的工程太过于复杂,从网上找了些文章,发现vs2017能够支持lin ...
- scikit-learn使用PCA降维小结
本文在主成分分析(PCA)原理总结和用scikit-learn学习主成分分析(PCA)的内容基础上做了一些笔记和补充,强调了我认为重要的部分,其中一些细节不再赘述. Jupiter notebook版 ...
- Django 使用 Celery 实现异步任务
对于网站来说,给用户一个较好的体验是很重要的事情,其中最重要的指标就是网站的浏览速度.因此服务端要从各个方面对网站性能进行优化,比如可采用CDN加载一些公共静态文件,如js和css:合并css或者js ...
- Base64编码图片存取与前台显示
需求:将Base64编码图片以BLOB类型存入数据库,需要时取出显示 后台: String base64str=new String(log.getRequest_imgdata());//log为实 ...
- Android:有关下拉菜单导航的学习(供自己参考)
Android:有关==下拉菜单导航==的学习 因为先前的学习都没想着记录自己的学习历程,所以该博客才那么迟才开始写. 内容: ==下拉菜单导航== 学习网站:android Spinner控件详解 ...
- int 和 Integer的区别
int是基本类型,默认值为0,int a=5;a只能用来计算,一般作为数值参数. Integer是引用类型,默认值为null, Integer b=5;b是一个对象,它可以有很多方法,一般做数值转换, ...
- ORM(object relational Maping)
ORM即对象关系映射,是一种为了解决面向对象与关系数据库存在的互不匹配的现象的技术. 简单的说,ORM是通过使用描述对象和数据库之间映射的元数据,将java程序中的对象自动持久化到关系数据库中.本质上 ...
- 四则运算3+psp0
题目要求: 1.程序可以判断用户的输入答案是否正确,如果错误,给出正确答案,如果正确,给出提示. 2.程序可以处理四种运算的混合算式. 3.要求两人合作分析,合作编程,单独撰写博客. 团队成员:张绍佳 ...