「日常训练」 Soldier and Traveling (CFR304D2E)
题意 (CodeForces 546E)
对一个无向图,给出图的情况与各个节点的人数/目标人数。每个节点的人只可以待在自己的城市或走到与他相邻的节点。
问最后是否有解,输出一可行解(我以为是必须和答案一样,然后本机调了半天死活不一样,交上去结果A了- -)。
分析
典型的网络流。问题在于建模。如何解决两个节点的人数->目标人数?
考虑到我们始终要和这两个状态打交道,不妨将每个结点拆成两个(转移前&转移后),这两个点间的流量是INF。如果两点相连(不妨设为u,u',v,v'),那么uu与v′v′、vv与u′u′分别有流量为INF的边。
转移前的点连接一个源点,其间的边流量为它们的人数,转移后的点连接一个汇点,边流量同理。
建模完后跑一遍最大流就有结果了,如果汇点没有那么多人那么无解,反之有解。
接下来的问题是如何求移动的人数。还记得增广路怎么求的吗?(紫书p367)其中,反向边的流量就是移动的具体数目。这样,我们就能够得出具体的移动方法了。
第一次写网络流,以后还要多加努力。
代码
#include <bits/stdc++.h>
#define MP make_pair
#define PB push_back
#define fi first
#define se second
#define ZERO(x) memset((x), 0, sizeof(x))
#define ALL(x) (x).begin(),(x).end()
#define rep(i, a, b) for (int i = (a); i <= (b); ++i)
#define per(i, a, b) for (int i = (a); i >= (b); --i)
#define QUICKIO \
ios::sync_with_stdio(false); \
cin.tie(0); \
cout.tie(0);
using namespace std;
using ll = long long;
using ull = unsigned long long;
using pi = pair<int,int>;
// 做题解
const int MAXN=1005;
//const int MAXM=3005;
const int INF=0x3f3f3f3f;
struct Edge
{
int from,to,cap,flow;
Edge(int f,int t,int c,int fl):from(f),to(t),cap(c),flow(fl) {}
};
struct Dinic
{
int n,m,s,t;
vector<Edge> edges;
vector<int> G[MAXN];
void AddEdge(int _f,int _t,int _c)
{
edges.PB(Edge(_f,_t,_c,0));
edges.PB(Edge(_t,_f,0,0));
m=edges.size();
G[_f].PB(m-2);
G[_t].PB(m-1);
}
bool vis[MAXN];
int d[MAXN],cur[MAXN];
bool BFS()
{
ZERO(vis);
queue<int> q;
q.push(s);
d[s]=0;
vis[s]=true;
while(!q.empty())
{
int x=q.front(); q.pop();
rep(i,0,G[x].size()-1)
{
Edge& e=edges[G[x][i]];
if(!vis[e.to] && e.cap>e.flow)
{
vis[e.to]=true;
d[e.to]=d[x]+1;
q.push(e.to);
}
}
}
return vis[t];
}
int DFS(int x,int a)
{
if(x==t||a==0) return a;
int flow=0,f;
for(int& i=cur[x]; i<G[x].size();++i)
{
Edge& e=edges[G[x][i]];
if(d[x]+1==d[e.to] && (f=DFS(e.to,min(a,e.cap-e.flow)))>0)
{
e.flow+=f;
edges[G[x][i]^1].flow-=f;
flow+=f;
a-=f;
if(a==0) break;
}
}
return flow;
}
int MaxFlow(int s,int t)
{
this->s=s; this->t=t;
int flow=0;
while(BFS())
{
ZERO(cur);
flow+=DFS(s,INF);
}
return flow;
}
};
int main()
{
Dinic dinic;
int n,m;
cin>>n>>m;
dinic.n=2*n+2;
//dinic.s=0; dinic.t=n+1;
ll ta=0,tb=0;
rep(i,1,n)
{
int tmp; cin>>tmp;
dinic.AddEdge(0,i,tmp);
ta+=tmp;
}
rep(i,1,n)
{
int tmp; cin>>tmp;
dinic.AddEdge(i+n,2*n+1,tmp);
tb+=tmp;
}
rep(i,1,n)
dinic.AddEdge(i,i+n,INF);
rep(i,1,m)
{
int a,b; cin>>a>>b;
dinic.AddEdge(a,b+n,INF);
dinic.AddEdge(b,a+n,INF);
}
int mf=dinic.MaxFlow(0,2*n+1);
if(mf==ta && mf==tb)
{
cout<<"YES"<<endl;
int f[105][105];
ZERO(f);
for(int i=0;i<dinic.m;i+=2)
{
int v=dinic.edges[i].to,u=dinic.edges[i^1].to;
//cout<<u<<" "<<v-n<<" "<<dinic.edges[i].flow<<endl;
if(u>=1 && u<=n)
{
f[u][v-n]=dinic.edges[i].flow;
}
}
rep(i,1,n)
{
rep(j,1,n)
{
cout<<string(j==1 ? "" : " ")<<f[i][j];
}
cout<<endl;
}
}
else cout<<"NO"<<endl;
return 0;
}
「日常训练」 Soldier and Traveling (CFR304D2E)的更多相关文章
- 「日常训练」 Soldier and Number Game (CFR304D2D)
题意 (Codeforces 546D) 给定一个数x=a!b!" role="presentation">x=a!b!x=a!b!的形式,问其中有几个质因数. 分 ...
- 「日常训练」 Soldier and Cards (CFR304D2C)
题意 (Codeforces 546C) 按照指定的规则打牌,问谁胜或无穷尽. 分析 又是一条模拟,用set+queue(这里手写了)处理即可.注意到两种局势"1 234"和&qu ...
- 「日常训练」Soldier and Badges (CFR304D2B)
题意 (Codeforces 546B) 问对一个序列最少需要增减几个1能使其彼此不同. 分析 模拟处理.需要注意的是,尽管题目中说了an<=3000,问题是,如果一群a全是3000呢(滑稽), ...
- 「日常训练」ZgukistringZ(Codeforces Round #307 Div. 2 B)
题意与分析(CodeForces 551B) 这他妈哪里是日常训练,这是日常弟中弟. 题意是这样的,给出一个字符串A,再给出两个字符串B,C,求A中任意量字符交换后(不限制次数)能够得到的使B,C作为 ...
- 「日常训练」 Fire!(UVA-11624)
与其说是训练不如说是重温.重新写了Java版本的代码. import java.util.*; import java.math.*; import java.io.BufferedInputStre ...
- 「日常训练」COMMON 约数研究(HYSBZ-1968)
题意与分析 感谢https://www.cnblogs.com/Leohh/p/7512960.html的题解.这题话说原来不在我的训练范围,正好有个同学问我,我就拿来做做.数学果然不是我擅长的啊,这 ...
- 「日常训练」 Mike and Fun (CFR305D2B)
题意(CodeForces 548B) 每次对01矩阵中的一位取反,问每次操作后,单列中最长连续1的长度. 分析 非常非常简单,但是我当时训练的时候WA了四次...无力吐槽了,人间 不值得.jpg 代 ...
- 「日常训练」Common Subexpression Elimination(UVa-12219)
今天做的题目就是抱佛脚2333 懂的都懂. 这条题目干了好几天,最后还是参考别人的代码敲出来了,但是自己独立思考了两天多,还是有收获的. 思路分析 做这条题我是先按照之前的那条题目(The SetSt ...
- 「日常训练」Magic Stones(CodeForces-1110E)
题意 给定两个数组c和t,可以对c数组中的任何元素变换\(c_i\)成\(c_{i+1}+c_{i-1}-c_i\),问c数组在若干次变换后能否变换成t数组. 分析 这种魔法题目我是同样的没做过. ...
随机推荐
- (第八场)G Counting regions 【欧拉公式】
题目链接:https://www.nowcoder.com/acm/contest/146/G G.Counting regions | 时间限制:1 秒 | 内存限制:128M Niuniu lik ...
- 2018.10.2 Eclipse中如何测地修改一个we项目步骤
找到项目的web.xml文件 大概的路径如下: 修改xml文件中的display-name 节点的值 下一步就是切换工作目录 显示的效果 打开最后一个文件修改 接下来找到这个文件 是部署的时候用的 运 ...
- JavaEE权限管理系统的搭建(六)--------使用拦截器实现菜单URL的跳转权限验证和页面的三级菜单权限按钮显示
本小结讲解,点击菜单进行页面跳转,看下图,点击管理员列表后会被认证拦截器首先拦截,验证用户是否登录,如果登录就放行,紧接着会被权限验证拦截器再次拦截,拦截的时候,会根据URL地址上找到对应的方法,然后 ...
- Object Detection with Discriminatively Trained Part Based Models
P. Felzenszwalb, R. Girshick, D. McAllester, D. RamananObject Detection with Discriminatively Traine ...
- 7.Vue-Quill-Editor图片插入自定义
Vue-Quill-Editor图片插入自定义 前言: 因为在项目中前端采用了Vue来实现,正好用到了富文本编辑器这一块,于是,经过技术上的选择,决定使用Vue-Quill-Editor. 使用的过程 ...
- 简析--Java中常见的一些关键字的解析
在Java开发中我们经常会用到一些关键字,关键字的定义很有意思"Java事先定义好的,具有特殊含义的单词",那么我们怎么来用好关键字呢?下面我们对一些常见的关键字进行分析和比较; ...
- 关于truthy 和 falsy
一,强制类型转换 JavaScript 在需要用到布尔类型值的上下文中使用强制类型转换(Type Conversion )将值转换为布尔值,比如:在条件语句或者循环语句中 一,truthy 在java ...
- 三种for循环遍历
import java.util.ArrayList;import java.util.Iterator;import java.util.List; public class For{ publi ...
- 利用nginx使ftp可以通过http访问
./nginx 启动服务./nginx -s stop 关闭服务./nginx -s reload 重新加载配置文件 搭建nginx映射ftp服务:打开nginx的配置文件nginx.conf(位于n ...
- Angular2入门学习
最近项目使用angular2,1和2版本变化大变样.下面总结一些学习网址及安装步骤. 中文官网(必看): https://angular.cn 懒人学习: http://www.imooc.com/l ...