题目链接

AtCoder:https://agc007.contest.atcoder.jp/tasks/agc007_e

洛谷:https://www.luogu.org/problemnew/show/AT2172

Solution

首先由于每条边只能经过两次,所以每次到了\(x\)点就会遍历完\(x\)子树再出来。

考虑二分答案,设当前二分的答案为\(mid\)。

设二元组\((a,b)_x\)表示\(x\)子树下第一次走代价为\(a\)最后一次为\(b\),且中间的过程都\(\leqslant \rm mid\)的一种方案。

那么显然可以得到一个暴力:

我们暴力枚举当前点左儿子和右儿子的二元组,设为\((a,b)_{ls},(c,d)_{rs}\),设左儿子的边权为\(x\),右儿子为\(y\),那么如果满足:\(b+c+x+y\leqslant {\rm mid}\),那么我们可以得到一个新的二元组\((a+x,d+y)_x\)。

考虑如何优化这个玩意,显然对于一个\(a\)只需要一个最小的\(b\),对于\(b\)也同理。

所以显然可以使用\(\rm two \ pointer\)做到每次合并为\(O(sz)\),其中\(sz\)为个数之和。

但是这样可以被卡成\(O(n^2)\),我们加一个启发式合并,每次合并小的,那么每次最多就是\(2|\min(S_{ls},S_{rs})|\)的,总复杂度\(O(n\log^2 n\log \rm ans)\),如果归并排序的话每次处理都是有序的,复杂度可以降为\(O(n\log n \log \rm ans)\)。

#include<bits/stdc++.h>
using namespace std; #define int long long void read(int &x) {
x=0;int f=1;char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if(ch=='-') f=-f;
for(;isdigit(ch);ch=getchar()) x=x*10+ch-'0';x*=f;
} void print(int x) {
if(x<0) putchar('-'),x=-x;
if(!x) return ;print(x/10),putchar(x%10+48);
}
void write(int x) {if(!x) putchar('0');else print(x);putchar('\n');} #define lf double
#define ll long long #define pii pair<int,int >
#define fr first
#define sc second
#define mp make_pair #define vec vector<pii >
#define pb push_back
#define iter vector <pii > :: iterator
#define for_vec(i,x) for(iter i=x.begin();i!=x.end();i++) #define _sort(x,y) sort(x.begin(),x.end(),y)
#define I (int) const int maxn = 3e5+10;
const int inf = 1e9;
const lf eps = 1e-8; int cmpx(pii x,pii y) {return x.fr<y.fr;}
int cmpy(pii x,pii y) {return x.sc<y.sc;} int son[maxn][2],v[maxn][2],n,d[maxn],mid,mn[maxn];
vec f[maxn]; void solve(vec &ls,vec &rs,vec &s) {
if(ls.size()>rs.size()) swap(ls,rs); //启发式合并
if(!s.empty()) s.clear();
_sort(ls,cmpy),_sort(rs,cmpx);
mn[0]=rs[0].sc;for(int i=1;i<I rs.size();i++) mn[i]=min(mn[i-1],rs[i].sc);
int p1=0,p2=rs.size()-1;
for(;p1<I ls.size();p1++) {
while(p2>=0&&ls[p1].sc+rs[p2].fr>mid) p2--;
if(~p2) s.pb(mp(ls[p1].fr,mn[p2]));
}
_sort(ls,cmpx),_sort(rs,cmpy);
mn[0]=rs[0].fr;for(int i=1;i<I rs.size();i++) mn[i]=min(mn[i-1],rs[i].fr);
for(p1=0,p2=rs.size()-1;p1<I ls.size();p1++) {
while(p2>=0&&ls[p1].fr+rs[p2].sc>mid) p2--;
if(~p2) s.pb(mp(mn[p2],ls[p1].sc));
}
} void dfs(int x) {
if(!son[x][0]) return f[x].pb(mp(0,0)),void();
int l=son[x][0],r=son[x][1];dfs(l),dfs(r);
vec &ls=f[l],&rs=f[r];
for_vec(i,ls) i->fr+=v[x][0],i->sc+=v[x][0];
for_vec(i,rs) i->fr+=v[x][1],i->sc+=v[x][1];
solve(ls,rs,f[x]);ls.clear(),rs.clear();
} signed main() {
read(n);
for(int i=2,x,y;i<=n;i++) read(x),read(y),son[x][d[x]]=i,v[x][d[x]++]=y;
int l=0,r=1e10,ans=1e10;
while(l<=r) {
mid=(l+r)>>1,dfs(1);
if(f[1].empty()) l=mid+1;
else r=mid-1,ans=mid;f[1].clear();
}write(ans);
return 0;
}

[AT2172] [agc007_e] Shik and Travel的更多相关文章

  1. 【AtCoder Grand Contest 007E】Shik and Travel [Dfs][二分答案]

    Shik and Travel Time Limit: 50 Sec  Memory Limit: 512 MB Description 给定一棵n个点的树,保证一个点出度为2/0. 遍历一遍,要求每 ...

  2. AGC007E Shik and Travel 解题报告

    AGC007E Shik and Travel 题目大意:\(n\) 个点的二叉树,每个点要么两个儿子,要么没有儿子,每条边有边权. 你从 \(1\) 号节点出发,走到一个叶子节点.然后每一天,你可以 ...

  3. AT2172 Shik and Travel

    题目描述: luogu 题解: 二分+暴力$vector$+$dfs$. 记录下所有可能的子树内合法方案,双指针+归并合并. 代码: #include<vector> #include&l ...

  4. AtCoder Grand Contest 007 E:Shik and Travel

    题目传送门:https://agc007.contest.atcoder.jp/tasks/agc007_e 题目翻译 现在有一个二叉树,除了叶子每个结点都有两个儿子.这个二叉树一共有\(m\)个叶子 ...

  5. AtCoder AGC007E Shik and Travel (二分、DP、启发式合并)

    题目链接 https://atcoder.jp/contests/agc007/tasks/agc007_e 题解 首先有个很朴素的想法是,二分答案\(mid\)后使用可行性DP, 设\(dp[u][ ...

  6. [AGC007E] Shik and Travel

    题目 给定一棵n节点的 以1为根的 满二叉树 (每个非叶子节点恰好有两个儿子)n−1 条边. 第ii条边连接 i+1号点 和 ai, 经过代价为vi设这棵树有m个叶子节点定义一次合法的旅行为:(1) ...

  7. [atAGC007E]Shik and Travel

    二分枚举答案,判定答案是否合法 贪心:每一个叶子只能经过一遍,因此叶子的顺序一定是一个dfs序,即走完一棵子树中的所有叶子才会到子树外 根据这个贪心可以dp,设$f[k][l][r]$表示仅考虑$k$ ...

  8. 贪心/构造/DP 杂题选做Ⅲ

    颓!颓!颓!(bushi 前传: 贪心/构造/DP 杂题选做 贪心/构造/DP 杂题选做Ⅱ 51. CF758E Broken Tree 讲个笑话,这道题是 11.3 模拟赛的 T2,模拟赛里那道题的 ...

  9. AtCoder Grand Contest 007

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

随机推荐

  1. 基于ejabberd简单实现xmpp群聊离线消息

    首先,xmpp服务器是基于ejabberd.离线消息模块是mod_interact,原地址地址:https://github.com/adamvduke/mod_interact: 修改后实现群聊离线 ...

  2. Linux☞权限数字表示法

    权限数字表示法: 1.Linux有三种访问权限: a.可读:r(Read) b.可写:w(Write) c.可执行:x(eXcute) 2.简单说说如何去看该文件的访问权限呢?一般我们执行命令,查看目 ...

  3. 03-运行第一个docker容器

    环境选择 容器需要管理工具.runtime 和操作系统,我们的选择如下: 1.管理工具 - Docker Engine因为 Docker 最流行使用最广泛. 2.runtime - runc Dock ...

  4. cinder创建volume的流程-简单梳理

    1. cinder-api接收到创建的请求,入口:cinder.api.v2.volumes.VolumeController#create,该方法主要负责一些参数的重新封装和校验,然后调用cinde ...

  5. 算法笔记(c++)--01背包问题

    算法笔记(c++)--经典01背包问题 算法解释起来太抽象了.也不是很好理解,最好的办法就是一步步写出来. 背包问题的核心在于m[i][j]=max(m[i-1][j],m[i-1][j-w[i]]+ ...

  6. 回归Qt——写在Qt5.10发布之日

    今天偶然看到一条关于Qt5.10发布的消息,发现Qt经历了诺基亚风波之后发展得依然良好,感到很欣慰.回头看上次关注Qt技术还是2011年,那时候用Qt4.7做一个小项目,对于一个写Win32界面和MF ...

  7. 转载笔记:DropDownList无限级分类(灵活控制显示形式)

    主要使用递归实现,数据库结构: 最终样式:  1protected void Page_Load(object sender, EventArgs e) 2    { 3        if (!Pa ...

  8. 手机上Uncaught ReferenceError: __WEBPACK_AMD_DEFINE_ARRAY__ is not defined的错误

    问题: vue2.0和webpack2.X的写的项目在电脑的浏览器上打开正常,在手机浏览器中访问页面的时候不能正常显示,空白. 通过chrome的真机调试chrome://inspect/#devic ...

  9. 由A到D中间可不止“B、C”

    在电子信息系统的学习中,我们或许早就被告知现实世界是模拟的,而数字化的模拟世界则越来越展现更多的风采.但是所谓的数字和模拟只是相对的而已,你可以把模拟量当做无穷数字量的组合,也可以把数字量当做具有不同 ...

  10. 20个常用Linux性能监控工具/命令

    20个常用Linux性能监控工具/命令 对于 Linux/Unix 系统管理员非常有用的并且最常用的20个命令行系统监视工具.这些命令可以在所有版本的 Linux 下使用去监控和查找系统性能的实际原因 ...