【2016北京集训测试赛】river

HINT
注意是全程不能经过两个相同的景点,并且一天的开始和结束不能用同样的交通方式。
[吐槽]
嗯。。看到这题的想法的话。。先想到了每个点的度为2,然后就有点不知所措了
隐隐约约想到了网络流,但并没有继续往下想了。。。
听完学长的讲评之后(%xj)个人觉得建图还是很有意思的ovo
[题解]
因为每个点到对面都有k种方式,那就想到每个点原来的点$x_0$拆成k个点$x_1$, $x_2$, $x_3$... $x_k$
然后很自然地$x_0$和拆成的点之间要连边
容量的话,因为hint里面的限制,也就是说一个点到另一个点的k中交通方式中只能选一种
(因为每个点只能到一次,而开始和结束不能用同样的方式)
这样一来容量显然就应该是1了
两岸之间的连接,就直接按照读入左岸连到右岸就好,容量也为1
(但其实因为左岸的流入流量和右岸的流出流量都有限制,中间的那条好像容量取1~ $\infty$都可以。。。%yxq)
接着考虑最后的答案是怎么得到的,会发现其实我们最后的到的路线是若干个环,每个点的度为2(一个大概长这样的)

如此一来,就会有个大胆的想法
对于每一个左岸的$x_0$,我们连一条源点到它的容量为2的边
对于每一个右岸的$x_0$,我们连一条它到汇点的容量为2的边
这样起到一个限制了每个点的度的作用,就可以保证有环并且环内每个点的度都为2(个人感觉这点是很有意思的)
于是乎最终的到的图长这样(以样例为例)

那么现在考虑构造方案
看回之前建图的思路,很容易得到的一个结论是满流的边肯定就是要走的边
那么现在问题就变成知道一堆边然后构造方案啦
很简单粗暴的方法直接强行把每个环走一遍记录下答案就好
天数的话就看有多少个环就好啦
挫挫的代码qwq
#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
#define inf 2147483647
using namespace std;
const int MAXN=**+;
struct xxx
{
int y,next,op,r,x;
}a[MAXN*];
queue<int> q;
int h[MAXN],lv[MAXN],id[][],num[MAXN];
int go[MAXN][],ans[][];
bool vis[MAXN];
int n,m,k,vs,vt,tot,tot1;
int add(int x,int y,int r);
int bfs();
int dfs(int v,int o);
int get_ans(); int main()
{
freopen("a.in","r",stdin); int x,y,z;
scanf("%d%d%d",&n,&m,&k);
memset(h,-,sizeof(h));
tot=;
vs=,vt=MAXN-;
for (int i=;i<=n+n;++i)
for (int j=;j<=k;++j)
id[i][j]=++tot,num[tot]=i;
tot=;
for (int i=;i<=n;++i)
{
add(vs,id[i][],);
add(id[i+n][],vt,);
for (int j=;j<=k;++j)
add(id[i][],id[i][j],),add(id[i+n][j],id[i+n][],);
}
for (int i=;i<=m;++i)
{
scanf("%d%d%d",&x,&y,&z);
add(id[x][z],id[y+n][z],);
}
while (bfs()) dfs(vs,inf);
get_ans();
} int add(int x,int y,int r)
{
a[++tot].y=y; a[tot].next=h[x]; h[x]=tot; a[tot].r=r; a[tot].op=tot+;
a[++tot].y=x; a[tot].next=h[y]; h[y]=tot; a[tot].r=; a[tot].op=tot-;
} int bfs()
{
while (!q.empty()) q.pop();
memset(lv,,sizeof(lv));
q.push(vs);
lv[vs]=;
int v,u;
while (!q.empty())
{
v=q.front(); q.pop();
for (int i=h[v];i!=-;i=a[i].next)
{
u=a[i].y;
if (lv[u]||!a[i].r) continue;
q.push(u);
lv[u]=lv[v]+;
if (u==vt) return true;
}
}
return false;
} int dfs(int v,int o)
{
if (v==vt||o==) return o;
int u,flow,ret=;
for (int i=h[v];i!=-;i=a[i].next)
{
u=a[i].y;
if (lv[u]!=lv[v]+) continue;
flow=dfs(u,min(a[i].r,o));
if (flow)
{
a[i].r-=flow;
a[a[i].op].r+=flow;
ret+=flow;
o-=flow;
if (!o) break;
}
}
return ret;
} int get_ans()
{
int x,y,pre;
//go[i]记录与i相连的两个点
for (int i=;i<=n;++i)
for (int j=;j<=k;++j)
for (int tmp=h[id[i][j]];tmp!=-;tmp=a[tmp].next)
{
if (a[tmp].r||a[tmp].y==id[i][]) continue;
y=num[a[tmp].y];
if (!go[i][]) go[i][]=y;
else go[i][]=y; if (!go[y][]) go[y][]=i;
else go[y][]=i;
}
memset(vis,false,sizeof(vis));
int cnt=;
for (int i=;i<=n;++i)
{
if (vis[i]) continue;
++cnt;
//将每个环走一遍
pre=i,x=go[i][];
ans[cnt][++ans[cnt][]]=i;
vis[i]=true;
while (x!=i)
{
vis[x]=true;
ans[cnt][++ans[cnt][]]=x;
if (pre==go[x][]) pre=x,x=go[x][];
else pre=x,x=go[x][];
}
ans[cnt][++ans[cnt][]]=x;
}
printf("%d\n",cnt);
//因为建图的方式所以左右岸肯定是交错来的
for (int i=;i<=cnt;++i)
{
printf("%d ",ans[i][]);
for (int j=;j<=ans[i][];++j)
if (j&) printf("L%d ",ans[i][j]);
else printf("R%d ",ans[i][j]-n);
printf("\n");
}
}
【2016北京集训测试赛】river的更多相关文章
- 2016北京集训测试赛(十六)Problem B: river
Solution 这题实际上并不是构造题, 而是一道网络流. 我们考虑题目要求的一条路径应该是什么样子的: 它是一个环, 并且满足每个点有且仅有一条出边, 一条入边, 同时这两条边的权值还必须不一样. ...
- 【2016北京集训测试赛(十六)】 River (最大流)
Description Special Judge Hint 注意是全程不能经过两个相同的景点,并且一天的开始和结束不能用同样的交通方式. 题解 题目大意:给定两组点,每组有$n$个点,有若干条跨组 ...
- 【2016北京集训测试赛(十)】 Azelso (期望DP)
Time Limit: 1000 ms Memory Limit: 256 MB Description 题解 状态表示: 这题的状态表示有点难想...... 设$f_i$表示第$i$个事件经过之 ...
- 【2016北京集训测试赛(二)】 thr (树形DP)
Description 题解 (这可是一道很早就碰到的练习题然后我不会做不想做,没想到在Contest碰到欲哭无泪......) 题目大意是寻找三点对的个数,使得其中的三个点两两距离都为d. 问题在于 ...
- 【2016北京集训测试赛(八)】 crash的数列 (思考题)
Description 题解 题目说这是一个具有神奇特性的数列!这句话是非常有用的因为我们发现,如果套着这个数列的定义再从原数列引出一个新数列,它居然还是一样的...... 于是我们就想到了能不能用多 ...
- 【2016北京集训测试赛】azelso
[吐槽] 首先当然是要orzyww啦 以及orzyxq奇妙顺推很强qwq 嗯..怎么说呢虽然说之前零零散散做了一些概d的题目但是总感觉好像并没有弄得比较明白啊..(我的妈果然蒟蒻) 这题的话可以说是难 ...
- [2016北京集训测试赛17]crash的游戏-[组合数+斯特林数+拉格朗日插值]
Description Solution 核心思想是把组合数当成一个奇怪的多项式,然后拉格朗日插值..:哦对了,还要用到第二类斯特林数(就是把若干个球放到若干个盒子)的一个公式: $x^{n}=\su ...
- [2016北京集训测试赛15]statement-[线段树+拆环]
Description Solution 由于题目要求,将a[i]->b[i](边权为i)后所得的图应该是由森林和环套树组合而成. 假如是树形结构,所有的t[i]就直接在线段树t[i]点的dfs ...
- [2016北京集训测试赛5]小Q与内存-[线段树的神秘操作]
Description Solution 哇真的异常服气..线段树都可以搞合并和拆分的啊orzorz.神的世界我不懂 Code #include<iostream> #include< ...
随机推荐
- egametang框架服务端运行流程
et框架的构建块主要由entity和componet组成,类似unity的组件.一个Entity可以挂载多个不同Component.Entity和Component的共同基类Disposer用于提供对 ...
- 存个emacs配置
emacs配置 (global-set-key [f9] 'compile-file) (global-set-key [f10] 'gud-gdb) (global-set-key (kbd &qu ...
- 彻底理解 Android 中的阴影
如果我们想创造更好的 Android App,我相信我们需要遵循 Material Design 的设计规范.一般而言,Material Design 是一个包含光线,材质和投影的三维环境.如果我们想 ...
- 从flexible.js引入高德地图谈起的移动端适配
曾几何时,前端还仅仅是PC端的.随着移动时代的兴起,h5及css3的推陈出新.前端的领域慢慢的由传统的pc端转入了移动端,这也导致了前端这一职业在风口的一段时间出尽了风头. 从开始的惶恐和无从下手,慢 ...
- java9 - 异常处理
Java异常 1.异常初见 System.out.println(1/0); 运行上面语句之后打印出: Exception in thread "main" java.lang.A ...
- Create小程序
我有时候喜欢直接用命令行创建.编译.执行java文件, 每次创建一个文件都要新建一个.java文件,然后再编辑.java文件加入类名,主函数…… 这些流程我有点厌倦,于是就编写了一个超级简单的自动创建 ...
- html5版 音乐播放器
html5版本音乐播放器,支持iOS设备,案例地址:http://www.xttblog.com/?p=1277 功能说明 支持iOS设备,但是iOS不支持自动下一曲,这是iOS本身限制,支持touc ...
- nyoj161 取石子 (四) 威佐夫博弈
思路:详细证明见博弈总结 如何判断威佐夫博弈的奇异局势? 对于状态(a, b),c = b - a,如果是奇异局势必定满足 a == c * (1+√5)/ 2. AC代码 #include < ...
- 吾八哥学Selenium(三):操作复选框checkbox/单选框radio的方法
复选框checkbox和单选框radio是web网站里经常会使用到的两个控件,那么在web自动化测试的时候如何利用Selenium来操作这俩控件呢?今天我们就来简单入门练习一下! html测试页面代码 ...
- 基于Python的Flask的开发实战(第一节Flask安装)
1.安装python虚拟环境 easy_install virtualenv easy_install pip cd /home/admin virtualenv flask-website sour ...