题意 (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)的更多相关文章

  1. 「日常训练」 Soldier and Number Game (CFR304D2D)

    题意 (Codeforces 546D) 给定一个数x=a!b!" role="presentation">x=a!b!x=a!b!的形式,问其中有几个质因数. 分 ...

  2. 「日常训练」 Soldier and Cards (CFR304D2C)

    题意 (Codeforces 546C) 按照指定的规则打牌,问谁胜或无穷尽. 分析 又是一条模拟,用set+queue(这里手写了)处理即可.注意到两种局势"1 234"和&qu ...

  3. 「日常训练」Soldier and Badges (CFR304D2B)

    题意 (Codeforces 546B) 问对一个序列最少需要增减几个1能使其彼此不同. 分析 模拟处理.需要注意的是,尽管题目中说了an<=3000,问题是,如果一群a全是3000呢(滑稽), ...

  4. 「日常训练」ZgukistringZ(Codeforces Round #307 Div. 2 B)

    题意与分析(CodeForces 551B) 这他妈哪里是日常训练,这是日常弟中弟. 题意是这样的,给出一个字符串A,再给出两个字符串B,C,求A中任意量字符交换后(不限制次数)能够得到的使B,C作为 ...

  5. 「日常训练」 Fire!(UVA-11624)

    与其说是训练不如说是重温.重新写了Java版本的代码. import java.util.*; import java.math.*; import java.io.BufferedInputStre ...

  6. 「日常训练」COMMON 约数研究(HYSBZ-1968)

    题意与分析 感谢https://www.cnblogs.com/Leohh/p/7512960.html的题解.这题话说原来不在我的训练范围,正好有个同学问我,我就拿来做做.数学果然不是我擅长的啊,这 ...

  7. 「日常训练」 Mike and Fun (CFR305D2B)

    题意(CodeForces 548B) 每次对01矩阵中的一位取反,问每次操作后,单列中最长连续1的长度. 分析 非常非常简单,但是我当时训练的时候WA了四次...无力吐槽了,人间 不值得.jpg 代 ...

  8. 「日常训练」Common Subexpression Elimination(UVa-12219)

    今天做的题目就是抱佛脚2333 懂的都懂. 这条题目干了好几天,最后还是参考别人的代码敲出来了,但是自己独立思考了两天多,还是有收获的. 思路分析 做这条题我是先按照之前的那条题目(The SetSt ...

  9. 「日常训练」Magic Stones(CodeForces-1110E)

    题意 给定两个数组c和t,可以对c数组中的任何元素变换\(c_i\)​成\(c_{i+1}+c_{i-1}-c_i\)​,问c数组在若干次变换后能否变换成t数组. 分析 这种魔法题目我是同样的没做过. ...

随机推荐

  1. Doubly Linked List

    Doubly Linked List Your task is to implement a double linked list. Write a program which performs th ...

  2. Cesium.js学习第二天(立方体)

    var viewer = new Cesium.Viewer('cs'); viewer.entities.add({//图标 position: Cesium.Cartesian3.fromDegr ...

  3. javascript操作Date对象

    Date 对象用于处理日期和时间. 创建 Date 对象的语法: var myDate=new Date() Date 对象会自动把当前日期和时间保存为其初始值. 参数形式有以下5种: new Dat ...

  4. Java8函数之旅 (七) - 函数式备忘录模式优化递归

    前言 在上一篇开始Java8之旅(六) -- 使用lambda实现Java的尾递归中,我们利用了函数的懒加载机制实现了栈帧的复用,成功的实现了Java版本的尾递归,然而尾递归的使用有一个重要的条件就是 ...

  5. Angularjs实例应用

    <!DOCTYPE html><html><head><meta http-equiv="Content-Type" content=&q ...

  6. VMware虚拟机修改BIOS启动项

    vmware默认是硬盘启动,要进bios里面设置成开机的启动顺序,要将光盘设置成第一启动项.但vm的开机画面比笔记本的还要快很多,基本都在1s内的,想进入 bios里面也有难度.. 对于网上说的开vm ...

  7. mobileeye

    if a human can drive a car based on vision alone – so can a computer. 但是目前哪家能做到?

  8. select 获取选中option的值方法,选中option方法

    options=$("#Select option:selected"); options.attr('name');options.val(); options.text(); ...

  9. 关于Echarts的原生js获取DOM元素与动态加载DOM元素的冲突问题

    1.前言: 最近在做的看板项目,因为需要循环加载后台数据,并且用Echarts做数据呈现,所以jQuery和angular等库统统靠边站,Echarts用的是原生js获取DOM元素,至于诸多不兼容等深 ...

  10. 【TOJ 5065】最长连续子序列(前缀和)

    Description 给定一系列非负整数,求最长的连续子序列,使其和是7的倍数. Input 第一行为正整数N(1<=N<=50000),接下来有N行,每行有一个非负整数,所有整数不大于 ...