题目传送门:https://agc007.contest.atcoder.jp/tasks/agc007_e

题目翻译

现在有一个二叉树,除了叶子每个结点都有两个儿子。这个二叉树一共有\(m\)个叶子,你需要从\(1\)号点出发,旅行\(m+1\)天后回到\(1\)号结点,其中前\(m\)天每天需要在叶子节点结束,并且每个叶子结点只允许被经过一次,每条边只允许被经过两次,边有边权,旅行的费用即为两点之间简单路径上权值之和,然后问你费用最大的一天最小可以是多少。第一天和最后一天免费。

题解

嗯,这又是一道题解题……

我们考虑二分答案,那么问题就转化成了怎么判断“每天花费都不超过\(limit\)是否可以完成旅途”。由于每条边只允许被经过两次 ,所以每个子树只允许被进出一次各一次。我们设\(f_{ijk}\)表示\(i\)号点子树第一次进去花费\(j\),出来花费\(k\),内部叶子节点走来走去不超过\(limit\)是否可以做到。因为边权过大,状态存不下来,所以我们用\(vector\)存三元组\(<i,j,k>\)来表示这个状态为真。当\(j\)相同时我们只存较小的\(k\),所以状态个数只会有\(NlogN\)个。然后每次由两个儿子转移更新自己可以用归并排序,按\(j\)从小打大排。

时间复杂度:\(O(NlogNlogANS)\)

空间复杂度:\(O(NlogN)\)

代码如下:

#include <vector>
#include <cstdio>
#include <algorithm>
using namespace std;
typedef long long ll;
typedef pair<ll,ll> pll;
typedef vector<pll> vp;
#define fr first
#define sc second const int maxn=131073; int n,tot;
ll dis[maxn];
int son[maxn][2],num[maxn];
ll l,r=1ll*maxn*maxn,limit; vp f[maxn]; int read() {
int x=0,f=1;char ch=getchar();
for(;ch<'0'||ch>'9';ch=getchar())if(ch=='-')f=-1;
for(;ch>='0'&&ch<='9';ch=getchar())x=x*10+ch-'0';
return x*f;
} vp dp(vp a,vp b) {
vp res;res.clear();
int x=a.size(),y=b.size(),st1=0,st2=0;
if(!y)return res;
while(st1!=x) {
while(st2!=y&&a[st1].sc+b[st2].fr<=limit)st2++;
if(st2)st2--;
if(a[st1].sc+b[st2].fr<=limit)res.push_back(make_pair(a[st1].fr,b[st2].sc));
st1++;
}
return res;
} void insert(vp &a,pll b) {
if(a.empty())a.push_back(b);
else if(a.back().fr==b.fr)a.back().sc=min(a.back().sc,b.sc);
else if(b.sc<a.back().sc)a.push_back(b);
} vp merge(vp a,vp b) {
vp res;res.clear();
int x=a.size(),y=b.size(),st1=0,st2=0;
while(st1!=x&&st2!=y) {
if(a[st1].fr<b[st2].fr)insert(res,a[st1++]);
else insert(res,b[st2++]);
}
while(st1!=x)insert(res,a[st1++]);
while(st2!=y)insert(res,b[st2++]);
return res;
} void dfs(int u) {
if(!num[u]) {
f[u].clear();
f[u].push_back(make_pair(dis[u],dis[u]));
return;
}
dfs(son[u][0]),dfs(son[u][1]);limit+=dis[u]*2;
vp tmp1=dp(f[son[u][0]],f[son[u][1]]);
vp tmp2=dp(f[son[u][1]],f[son[u][0]]);
f[u]=merge(tmp1,tmp2);
limit-=dis[u]*2;
} bool check() {
dfs(1);
if(!f[1].empty())return 1;
return 0;
} int main() {
n=read();
for(int i=2;i<=n;i++) {
int fa=read(),v=read();
dis[i]=dis[fa]+v;
son[fa][num[fa]++]=i;
}
while(l<r) {
limit=(l+r)>>1;
if(check())r=limit;
else l=limit+1;
}
printf("%lld\n",r);
return 0;
}

AtCoder Grand Contest 007 E:Shik and Travel的更多相关文章

  1. AtCoder Grand Contest 007

    AtCoder Grand Contest 007 A - Shik and Stone 翻译 见洛谷 题解 傻逼玩意 #include<cstdio> int n,m,tot;char ...

  2. AtCoder Grand Contest 009 D:Uninity

    题目传送门:https://agc009.contest.atcoder.jp/tasks/agc009_d 题目翻译 定义只有一个点的树权值为\(0\),若干棵(可以是\(0\)棵)权值为\(k\) ...

  3. AtCoder Grand Contest 009 E:Eternal Average

    题目传送门:https://agc009.contest.atcoder.jp/tasks/agc009_e 题目翻译 纸上写了\(N\)个\(1\)和\(M\)个\(0\),你每次可以选择\(k\) ...

  4. AtCoder Grand Contest 004 C:AND Grid

    题目传送门:https://agc004.contest.atcoder.jp/tasks/agc004_c 题目翻译 给你一张网格图,指定的格子是紫色的,要求你构造出两张网格图,其中一张你可以构造一 ...

  5. AtCoder Grand Contest 014 D:Black and White Tree

    题目传送门:https://agc014.contest.atcoder.jp/tasks/agc014_d 题目翻译 给你一棵树,每次任选一个点染色,先手染白色,后手染黑色.如果最后存在一个白色的点 ...

  6. AtCoder Grand Contest 028 A:Two Abbreviations

    题目传送门:https://agc028.contest.atcoder.jp/tasks/agc028_a 题目翻译 给你两个串\(s\)与\(t\),长度分别为\(n,m\).问你存不存在一个串长 ...

  7. AtCoder Grand Contest 002 F:Leftmost Ball

    题目传送门:https://agc002.contest.atcoder.jp/tasks/agc002_f 题目翻译 你有\(n*k\)个球,这些球一共有\(n\)种颜色,每种颜色有\(k\)个,然 ...

  8. AtCoder Grand Contest 013 C:Ants on a Circle

    题目传送门:https://agc013.contest.atcoder.jp/tasks/agc013_c 题目翻译 给你一个周长为\(L\)的圆,有\(N\)只蚂蚁在圆上爬,速度为一单位距离每秒. ...

  9. AtCoder Grand Contest 010 C:Cleaning

    题目传送门:https://agc010.contest.atcoder.jp/tasks/agc010_c 题目翻译 给你一棵树,每个点有个权值,每次操作可以选择两个度数为\(1\)的结点,然后让这 ...

随机推荐

  1. 页面登陆框老是乱乱的?banner跨页图片缩小之后总是在側面不能显示主要部分?哈哈~我来帮你忙~~

    有banner背景图片和登陆框的html.css排布 目的:无论页面大小,背景图片都要居中(显示图片中间主要内容,而不是側面的一些东西),登陆框基本能在页面内显示. 盒子的排列应该是这种: <d ...

  2. java内存模型(Java Memory Model)

    内容导航: Java内存模型 硬件存储体系结构 Java内存模型和硬件存储体系之间的桥梁: 共享对象的可见性 竞争条件 Java内存模型规定了JVM怎样与计算机存储系统(RAM)协调工作.JVM是一个 ...

  3. 在kubernetes 集群运行 odoo

        kubernetes 可以自动运行多个 odoo服务的副本,因此 非常适用用来做高可用的odoo部署, 在本例中,odoo服务运行在 kubernetes 集群中, 而 postgreSQL ...

  4. meta标签多种用法

    <meta name=”google” content=”notranslate” /> <!-- 有时,Google在结果页面会提供一个翻译链接,但有时候你不希望出现这个链接,你可 ...

  5. 常见Struts、Hibernate、Spring、J2EE、ibatis、Oracle等开发框架架构图及其简介

    各种系统架构图及其简介 转载请保留出处,不胜人生一场醉汇总. 以下文字和架构图均在本人相关系统设计和架构方案中有所应用. 原文出处:http://space.itpub.net/6517/viewsp ...

  6. HDU 5667 :Sequence

    Sequence  Accepts: 59  Submissions: 650  Time Limit: 2000/1000 MS (Java/Others)  Memory Limit: 65536 ...

  7. kubernetes调度之资源配额示例

    系列目录 前面说过,资源配额限制在指定名称空间下,对资源对象数量和特定类型的资源的限制,你可以在ResourceQuota中指定配额 创建名称空间 我们创建一个新的名称空间来演示 kubectl cr ...

  8. npm 淘宝设置代理

    直接安装cnpm导致无限索引,因此直接使用代理 方法一: 直接在当前用户文件夹下,npmrc 文件上直接设置代理:registry=https://registry.npm.taobao.org 方法 ...

  9. JS 省市两级联动(不带地区版本)

    基于网上找的一个版本改造,因为项目需求不需要地区只要省.市,所以做了改版,两个input上直接取出了数据 <html> <head> <script src=" ...

  10. jave 金额科学记数法处理

    金额 :amount amount.stripTrailingZeros().toPlainString();