【BZOJ4849】[Neerc2016]Mole Tunnels

Description

鼹鼠们在底下开凿了n个洞,由n-1条隧道连接,对于任意的i>1,第i个洞都会和第i/2(取下整)个洞间有一条隧道,第i个洞内还有ci个食物能供最多ci只鼹鼠吃。一共有m只鼹鼠,第i只鼹鼠住在第pi个洞内,一天早晨,前k只鼹鼠醒来了,而后n-k只鼹鼠均在睡觉,前k只鼹鼠就开始觅食,最终他们都会到达某一个洞,使得所有洞的ci均大于等于该洞内醒着的鼹鼠个数,而且要求鼹鼠行动路径总长度最小。现对于所有的1<=k<=m,输出最小的鼹鼠行动路径的总长度,保证一定存在某种合法方案。

Input

第一行两个数n,m(1<=n,m<=100000),表示有n个洞,m只鼹鼠。
第二行n个整数ci表示第i个洞的食物数。
第三行m个整数pi表示第i只鼹鼠所在洞pi。

Output

输出一行m个整数,第i个整数表示当k=i时最小的鼹鼠行动路径总长度。

Sample Input

5 4 0 0 4 1 1 2 4 5 2

Sample Output

1 1 2 4

题解:一眼看到这道题,直接想到费用流,但是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 模拟费用流的更多相关文章

  1. BZOJ4849[Neerc2016]Mole Tunnels——模拟费用流+树形DP

    题目描述 鼹鼠们在底下开凿了n个洞,由n-1条隧道连接,对于任意的i>1,第i个洞都会和第i/2(取下整)个洞间有一条隧 道,第i个洞内还有ci个食物能供最多ci只鼹鼠吃.一共有m只鼹鼠,第i只 ...

  2. BZOJ 4849 [NEERC2016]Mole Tunnels (模拟费用流)

    题目链接 https://www.lydsy.com/JudgeOnline/problem.php?id=4849 题解 其实也是模拟费用流,但是这道题和一般的题目不一样,这道题是在一个完全二叉树上 ...

  3. [bzoj4849][Neerc2016]Mole Tunnels

    来自FallDream的博客,未经允许,请勿转载,谢谢 貌似是省队集训女队讲的题... 今天在bzoj找一道题无果,但是翻到了这道就顺便写了下. 鼹鼠们在底下开凿了n个洞,由n-1条隧道连接,对于任意 ...

  4. P6122-[NEERC2016]Mole Tunnels【模拟费用流】

    正题 题目链接:https://www.luogu.com.cn/problem/P6122 题目大意 给出\(n\)个点的一棵满二叉树,每个点有容量\(c_i\),\(m\)次从\(p_i\)处加一 ...

  5. 贪心(模拟费用流):NOIP2011 观光公交

    [问题描述] 风景迷人的小城Y 市,拥有n 个美丽的景点.由于慕名而来的游客越来越多,Y 市特意安排了一辆观光公交车,为游客提供更便捷的交通服务.观光公交车在第0 分钟出现在1号景点,随后依次前往2. ...

  6. BZOJ4977[Lydsy1708月赛]跳伞求生——贪心+堆+模拟费用流

    题目链接: 跳伞求生 可以将题目转化成数轴上有$n$个人和$m$个房子,坐标分别为$a_{i}$和$b_{i}$,每个人可以进一个他左边的房子,每个房子只能进一个人.每个房子有一个收益$c_{i}$, ...

  7. 【bzoj1150】[CTSC2007]数据备份Backup 模拟费用流+链表+堆

    题目描述 你在一家 IT 公司为大型写字楼或办公楼(offices)的计算机数据做备份.然而数据备份的工作是枯燥乏味的,因此你想设计一个系统让不同的办公楼彼此之间互相备份,而你则坐在家中尽享计算机游戏 ...

  8. [UOJ455][UER #8]雪灾与外卖——堆+模拟费用流

    题目链接: [UOJ455]雪灾与外卖 题目描述:有$n$个送餐员(坐标为$x_{i}$)及$m$个餐厅(坐标为$y_{i}$,权值为$w_{i}$),每个送餐员需要前往一个餐厅,每个餐厅只能容纳$c ...

  9. 【CF280D】 k-Maximum Subsequence Sum ,线段树模拟费用流

    昨天考试被教育了一波.为了学习一下\(T3\)的科技,我就找到了这个远古时期的\(cf\)题(虽然最后\(T3\)还是不会写吧\(QAQ\)) 顾名思义,这个题目其实可以建成一个费用流的模型.我们用流 ...

随机推荐

  1. Laravel 学习 .env文件 getenv 获得环境变量的值

    Laravel 学习 .env文件 getenv 获得环境变量的值  我们还需要对应用的 .env 文件进行设置,为应用指定数据库名称 sample. .env . . . DB_DATABASE=s ...

  2. 去除前后空格,Oracle和SQLSERVER都适用。ltrim(rtrim(’ ‘))

    Oracle自带去除方法:trim(). 但是sql语法中没有直接去除两头空格的函数,但有ltrim()去除左空格rtrim()去除右空格. 合起来用就是sql的trim()函数,即select lt ...

  3. Nginx Errors: upstream response cache error

    Nginx Errors upstream response cache error *2470578 an upstream response is buffered to a temporary ...

  4. iOS开发-项目的完整重命名方法,图文教程。

    前言:在IOS开发中,有时候想改一下项目的名字,都会遇到很多麻烦.直接改项目名吧,XCODE又不会帮你改所有的名字.总是有很多文件.文件夹或者是项目设置的项.而且都是不能随便改的,有时候改着改着,编译 ...

  5. 牛散NO.3:MACD放之四海 假作真时真亦假

    大宗商品日线“异曲同工夺命勾魂枪” 话说有实战意义的技术在任何资本市场里都能产生出神奇的效果.不能说放之四海皆准,但至少起到触类旁通的“牵强”吧.大宗商品特别是在国际市场交易的大宗 商品由于是来自各方 ...

  6. ARM(CM3)的汇编指令

    转http://blog.csdn.net/gaojinshan/article/details/11534569 16位数据操作指令 名字 功能ADC  带进位加法(ADD with Carry)  ...

  7. Predix中模型设计

    GE的Predix使用了图形数据库作为Asset存储,用以解决传统RDBMS系统中扩展性差,不支持行的动态定义问题. 对于实体,或者Asset来说,由如下几个方面表述: ID: Ties everyt ...

  8. 更新换代----systemctl命令取代chkconfig和service

    systemctl命令是系统服务管理器指令,它实际上将 service 和 chkconfig 这两个命令组合到一起. 任务 旧指令 新指令 使某服务自动启动 chkconfig --level 3 ...

  9. stick footers布局

    需求: 将footer固定到底部.文章内容不足满屏时 footer在底部,超过满屏时footer在内容末尾. 方法一: <div id="wrap"> <div ...

  10. 微信强大的demo

    https://github.com/xialeistudio/wechatPublicPlatform