【BZOJ4849】[Neerc2016]Mole Tunnels 模拟费用流
【BZOJ4849】[Neerc2016]Mole Tunnels
Description
Input
Output
Sample Input
Sample Output
题解:一眼看到这道题,直接想到费用流,但是n太大了,不能直接跑费用流,但是我们发现题中给的是一棵完全二叉树,所以我们可以想一想怎么利用这个性质。
我们依旧沿用费用流的思想,假设我们扫到了第i只鼹鼠,我们先找到距离i最近的有空位置的点,然后在将那个点的剩余空间-1,再讲路径上所有的反向边流量+1(因为正向边的流量可以看做正无穷,所以我们不管)。现在我们的思路就很清晰了,所以我们只要找到距离i最近的有空位置的点就行了。
具体做法是先扫一遍整棵树,用f[i]表示在i的子树中距离i最近的点到i的距离,g[i]表示在i的子树中距离i最近的点的位置。查询的时候,从p[i]一步一步向上爬,不断用当前节点的 f值+爬上来的距离 更新答案,最后暴力重构路径上的所有节点的f,g值,因为保证是完全二叉树,所以复杂度是O(nlogn)的。
说着挺容易其实代码挺丑的~
#include <cstdio>
#include <cstring>
#include <iostream>
#define lson x<<1
#define rson x<<1|1
using namespace std;
const int maxn=100010;
typedef long long ll;
ll ans;
int n,m,sum,minn,pos,lca;
int p[maxn],c[maxn],g[maxn],f[maxn],v[maxn][2]; //0上1下
int main()
{
scanf("%d%d",&n,&m);
int i,j;
for(i=1;i<=n;i++)
scanf("%d",&c[i]);
memset(f,0x3f,sizeof(f));
for(i=n;i>=1;i--)
{
if(c[i]) f[i]=0,g[i]=i;
if(f[i>>1]>f[i]+1) f[i>>1]=f[i]+1,g[i>>1]=g[i];
}
for(i=1;i<=m;i++)
{
scanf("%d",&p[i]);
minn=1<<30,sum=0;
for(j=p[i];j;j>>=1)
{
if(f[j]+sum<minn) minn=f[j]+sum,pos=g[j],lca=j;
sum+=(v[j][0]>0)?(-1):1;
}
ans=ans+minn;
printf("%lld",ans);
if(i!=m) printf(" ");
c[pos]--;
for(j=p[i];j!=lca;j>>=1) v[j][0]?v[j][0]--:v[j][1]++;
for(j=pos;j!=lca;j>>=1) v[j][1]?v[j][1]--:v[j][0]++;
for(j=p[i];j!=lca;j>>=1)
{
f[j]=1<<30;
if(c[j]&&0<f[j]) f[j]=0,g[j]=j;
if((j<<1)<=n&&f[j<<1]+(v[j<<1][1]?-1:1)<f[j]) f[j]=f[j<<1]+(v[j<<1][1]?-1:1),g[j]=g[j<<1];
if((j<<1|1)<=n&&f[j<<1|1]+(v[j<<1|1][1]?-1:1)<f[j]) f[j]=f[j<<1|1]+(v[j<<1|1][1]?-1:1),g[j]=g[j<<1|1];
}
for(j=pos;j;j>>=1)
{
f[j]=1<<30;
if(c[j]&&0<f[j]) f[j]=0,g[j]=j;
if((j<<1)<=n&&f[j<<1]+(v[j<<1][1]?-1:1)<f[j]) f[j]=f[j<<1]+(v[j<<1][1]?-1:1),g[j]=g[j<<1];
if((j<<1|1)<=n&&f[j<<1|1]+(v[j<<1|1][1]?-1:1)<f[j]) f[j]=f[j<<1|1]+(v[j<<1|1][1]?-1:1),g[j]=g[j<<1|1];
}
}
return 0;
}
【BZOJ4849】[Neerc2016]Mole Tunnels 模拟费用流的更多相关文章
- BZOJ4849[Neerc2016]Mole Tunnels——模拟费用流+树形DP
题目描述 鼹鼠们在底下开凿了n个洞,由n-1条隧道连接,对于任意的i>1,第i个洞都会和第i/2(取下整)个洞间有一条隧 道,第i个洞内还有ci个食物能供最多ci只鼹鼠吃.一共有m只鼹鼠,第i只 ...
- BZOJ 4849 [NEERC2016]Mole Tunnels (模拟费用流)
题目链接 https://www.lydsy.com/JudgeOnline/problem.php?id=4849 题解 其实也是模拟费用流,但是这道题和一般的题目不一样,这道题是在一个完全二叉树上 ...
- [bzoj4849][Neerc2016]Mole Tunnels
来自FallDream的博客,未经允许,请勿转载,谢谢 貌似是省队集训女队讲的题... 今天在bzoj找一道题无果,但是翻到了这道就顺便写了下. 鼹鼠们在底下开凿了n个洞,由n-1条隧道连接,对于任意 ...
- P6122-[NEERC2016]Mole Tunnels【模拟费用流】
正题 题目链接:https://www.luogu.com.cn/problem/P6122 题目大意 给出\(n\)个点的一棵满二叉树,每个点有容量\(c_i\),\(m\)次从\(p_i\)处加一 ...
- 贪心(模拟费用流):NOIP2011 观光公交
[问题描述] 风景迷人的小城Y 市,拥有n 个美丽的景点.由于慕名而来的游客越来越多,Y 市特意安排了一辆观光公交车,为游客提供更便捷的交通服务.观光公交车在第0 分钟出现在1号景点,随后依次前往2. ...
- BZOJ4977[Lydsy1708月赛]跳伞求生——贪心+堆+模拟费用流
题目链接: 跳伞求生 可以将题目转化成数轴上有$n$个人和$m$个房子,坐标分别为$a_{i}$和$b_{i}$,每个人可以进一个他左边的房子,每个房子只能进一个人.每个房子有一个收益$c_{i}$, ...
- 【bzoj1150】[CTSC2007]数据备份Backup 模拟费用流+链表+堆
题目描述 你在一家 IT 公司为大型写字楼或办公楼(offices)的计算机数据做备份.然而数据备份的工作是枯燥乏味的,因此你想设计一个系统让不同的办公楼彼此之间互相备份,而你则坐在家中尽享计算机游戏 ...
- [UOJ455][UER #8]雪灾与外卖——堆+模拟费用流
题目链接: [UOJ455]雪灾与外卖 题目描述:有$n$个送餐员(坐标为$x_{i}$)及$m$个餐厅(坐标为$y_{i}$,权值为$w_{i}$),每个送餐员需要前往一个餐厅,每个餐厅只能容纳$c ...
- 【CF280D】 k-Maximum Subsequence Sum ,线段树模拟费用流
昨天考试被教育了一波.为了学习一下\(T3\)的科技,我就找到了这个远古时期的\(cf\)题(虽然最后\(T3\)还是不会写吧\(QAQ\)) 顾名思义,这个题目其实可以建成一个费用流的模型.我们用流 ...
随机推荐
- 微信小程序flex容器属性详解
flex容器属性详解 flex-direction决定元素的排列方向 flex-wrap决定元素如何换行 flex-flow 是 flex-direction 和flex-wrap的简写 justif ...
- 异步编程C#回调方法
1.什么是异步? 异步操作通常用于执行完成时间可能较长的任务,如打开大文件.连接远程计算机或查询数据库.异步操作在主应用程序线程以外的线程中执行.应用程序调用方法异步执行某个操作时,应用程序可在异步方 ...
- ansible远程切换用户执行命令
ansible test -l 10.0.10.1 -e "ansible_become_user=www" -m shell -a "/data/publish/pub ...
- 一个关于git push失败的解决方案
问题背景:在GitHub上创建了一个repositorie, 本地初始化并添加了远程仓库后,在GitHub上创建了一个README.md文件(注意不是从本地git push上去的),随后本地修改工程源 ...
- 深入理解get和post的区别
GET和POST是HTTP请求的两种基本方法,要说它们的区别,接触过WEB开发的人都能说出一二.最直观的区别就是GET把参数包含在URL中,POST通过request body传递参数. 正常GET和 ...
- Sql中的内连接,左连接以及右连接区别
转自:http://pangaoyuan.javaeye.com/blog/713177 有两个表A和表B. 表A结构如下: Aid:int:标识种子,主键,自增ID Aname:varchar 数据 ...
- Xilinx 7系列例化MIG IP core DDR3读写
昨晚找了一下,发现DDR3读写在工程上多是通过例化MIG,调用生成IPcore的HDL Functional Model.我说嘛,自己哪能写出那么繁琐的,不过DDR读写数据可以用到状态机,后期再添砖加 ...
- [HNOI2008]玩具装箱toy(dp+斜率优化)
斜率优化问题一般都是决策单调问题.对于这题能够证明单调决策. 令sum[i]=sigma(c [k] ) 1<=k<=i , f[i]=sum[i]+i , c=L+1; 首先我们能 ...
- 时间序列 R 读书笔记 04 Forecasting: principles and practice
本章開始学习<Forecasting: principles and practice> 1 getting started 1.1 事件的可预言性 一个时间能不能被预言主要取决于以下三点 ...
- Creating Context Menu / 创建上下文菜单项 / VC++, Windows, DLL, ATL, COM
创建上下文菜单项 1.新建一个ATL Project. 2.建议将Project Property中Linker – General - “Register Output” 设为no,C/C++ - ...