题解 Children Trips
Description
给出一个大小为 \(n\) 的边权全为 \(1,2\) 的带权树,有 \(q\) 此查询,每次给出 \(u,v,p\) ,问 \(u\to v\) 每次可以最多走边权和 \(\le p\) 的路径,问最少走多少次。
\(n,q\le 10^5\)
Solution
因为自己没有想出来,所以还是写一发题解。
首先边权比较有迷惑性,但是这个 \(\le 2\) 确实多少用。考虑根号分治,可以发现 \(p>\sqrt n\) 最多只会爬 \(\sqrt n\) 次,所以可以直接暴力爬,即倍增找到每次可以走到的下一个点。
考虑 \(\sqrt n\),发现我们仍可以使用倍增,即处理出一个点走 \(2^i\) 次会走到哪里。
复杂度 \(\Theta(n\sqrt n\log n)\) 。
Code
#include <bits/stdc++.h>
using namespace std;
#define Int register int
#define MAXN 100005
template <typename T> inline void read (T &t){t = 0;char c = getchar();int f = 1;while (c < '0' || c > '9'){if (c == '-') f = -f;c = getchar();}while (c >= '0' && c <= '9'){t = (t << 3) + (t << 1) + c - '0';c = getchar();} t *= f;}
template <typename T,typename ... Args> inline void read (T &t,Args&... args){read (t);read (args...);}
template <typename T> inline void write (T x){if (x < 0){x = -x;putchar ('-');}if (x > 9) write (x / 10);putchar (x % 10 + '0');}
template <typename T> inline void chkmax (T &a,T b){a = max (a,b);}
template <typename T> inline void chkmin (T &a,T b){a = min (a,b);}
int n,m,siz;
struct edge{int v,w;};
vector <edge> g[MAXN];
struct str{
int u,v,P,ind;
bool operator < (const str &t)const{return P < t.P;}
}seq[MAXN];
int dep[MAXN],dis[MAXN],par[MAXN][21];
int getlca (int u,int v){
if (dep[u] < dep[v]) swap (u,v);
for (Int i = 20,dis = dep[u] - dep[v];~i;-- i) if (dis >> i & 1) u = par[u][i];
if (u == v) return u;
else{
for (Int i = 20;~i;-- i) if (par[u][i] ^ par[v][i]) u = par[u][i],v = par[v][i];
return par[u][0];
}
}
void dfs (int u,int fa){
dep[u] = dep[fa] + 1,par[u][0] = fa;
for (Int i = 1;i <= 20;++ i) par[u][i] = par[par[u][i - 1]][i - 1];
for (edge to : g[u]){
int v = to.v,w = to.w;
if (v == fa) continue;
dis[v] = dis[u] + w,dfs (v,u);
}
}
int getjump (int u,int p){
for (Int i = 20;~i;-- i) if (par[u][i] && dis[u] - dis[par[u][i]] <= p)
// cout << u << " -> " << par[u][i] << endl,
p -= dis[u] - dis[par[u][i]],u = par[u][i];
return u;
}
struct node{int lft,stp;};
node getit1 (int u,int lca,int p){
int stp = 0;
while (u ^ lca){
int now = getjump (u,p);
if (dep[now] > dep[lca]) stp ++,u = now;
else break;
}
return node{dis[u] - dis[lca],stp};
}
int st[MAXN][21];
void makeit (int P){
for (Int i = 1;i <= n;++ i) st[i][0] = getjump (i,P);
// cout << P << ": ------------ " << endl;
for (Int j = 1;(1 << j) <= n;++ j)
for (Int i = 1;i <= n;++ i)
st[i][j] = st[st[i][j - 1]][j - 1];
// cout << i << " jump " << (1 << j) << ": " << st[i][j] << endl;
}
node getit2 (int u,int lca,int p){
int stp = 0;
for (Int i = 20;~i;-- i) if (st[u][i] && dep[st[u][i]] > dep[lca]) stp += (1 << i),u = st[u][i];
return node{dis[u] - dis[lca],stp};
}
int solveit (int x,int y,int P){
if (x == y) return 0;
int lca = getlca (x,y);
node t1 = P <= siz ? getit2 (x,lca,P) : getit1 (x,lca,P);
node t2 = P <= siz ? getit2 (y,lca,P) : getit1 (y,lca,P);
int now = t1.stp + t2.stp;
if (t1.lft + t2.lft <= P) return now + 1;
else return now + 2;
}
int ans[MAXN];
signed main(){
// freopen ("data.in","r",stdin);
// freopen ("f1.out","w",stdout);
read (n),siz = sqrt (n);
for (Int i = 2,u,v,w;i <= n;++ i) read (u,v,w),g[u].push_back (edge{v,w}),g[v].push_back (edge{u,w});
dfs (1,0),read (m);
for (Int i = 1;i <= m;++ i) read (seq[i].u,seq[i].v,seq[i].P),seq[i].ind = i;
sort (seq + 1,seq + m + 1);
for (Int i = 1;i <= m;++ i){
if (seq[i].P <= siz){
if (seq[i].P != seq[i - 1].P) makeit (seq[i].P);
ans[seq[i].ind] = solveit (seq[i].u,seq[i].v,seq[i].P);
}
else ans[seq[i].ind] = solveit (seq[i].u,seq[i].v,seq[i].P);
}
for (Int i = 1;i <= m;++ i) write (ans[i]),putchar ('\n');
return 0;
}
题解 Children Trips的更多相关文章
- [CC-TRIPS]Children Trips
[CC-TRIPS]Children Trips 题目大意: \(n(n\le10^5)\)座城市构成一棵树,且树上的每条边的长度\(l_i\)满足\(1\le l_i\le 2\).\(m(m\le ...
- 【codechef】Children Trips
Portal -->CC_Children Trips Solution (英文题解看得真爽qwq不过写的好详细啊ovo) 首先这题有一个很重要的条件就是边权是\(1\)或者\(2\),所以虽然 ...
- CODECHEF Oct. Challenge 2014 Children Trips
@(XSY)[分塊, 倍增] Description There's a new trend among Bytelandian schools. The "Byteland Tourist ...
- Codechef TRIPS Children Trips (分块、倍增)
题目链接: https://www.codechef.com/problems/TRIPS 感觉CC有点毒瘤啊.. 题解: 首先有一个性质可能是因为太傻所以网上没人解释,然而我看了半天: 就是正序和倒 ...
- 【CODECHEF】Children Trips 倍增
此题绝了,$O(n^{1.5}\ log\ n)$都可以过掉.... 题目大意:给你一颗$n$个点的树,每条边边权不是2就是$1$,有$m$个询问,每次询问一个人从$x$点走到$y$点,每天可以走的里 ...
- HDU1297 Children’s Queue (高精度+递推)
Children’s Queue Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) ...
- 《ACM国际大学生程序设计竞赛题解Ⅰ》——基础编程题
这个专栏开始介绍一些<ACM国际大学生程序设计竞赛题解>上的竞赛题目,读者可以配合zju/poj/uva的在线测评系统提交代码(今天zoj貌似崩了). 其实看书名也能看出来这本书的思路,就 ...
- To Miss Our Children Time(dp)
To Miss Our Children Time Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65768/65768 K (Jav ...
- CF1037E. Trips
题目链接 CF1037E. Trips 题解 每次删点后,对不满足要求的点拓扑 代码 #include<map> #include<queue> #include<vec ...
随机推荐
- assign()与create()的区别
Q:assign()与create()的区别? A:let obj = Object.assign(targetObj, -sourceObj) 作用:将一个或多个源对象自身的可枚举属性与目标对象的属 ...
- 高德地图——2D转换3D
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <script ty ...
- 🏆【JVM技术专区】「难点-核心-遗漏」TLAB内存分配+锁的碰撞(技术串烧)!
JVM内存分配及申请过程 当使用new关键字或者其他任何方式进行创建一个类的对象时,JVM虚拟机需要为该对象分配内存空间,而对象的大小在类加载完成后已经确定了,所以分配内存只需要在Java堆中划分出一 ...
- Spring Boot +Vue 项目实战笔记(三):数据库的引入
这一篇的主要内容是引入数据库并实现通过数据库验证用户名与密码. 一.引入数据库 之前说过数据库的采用是 MySQL,算是比较主流的选择,从性能和体量等方面都比较优秀,当然也有一些弊端,但数据库不是我们 ...
- vue + WangEnduit
components 注册组件 <template lang="html"> <div class="editor"> <div ...
- WebService学习总结(二)--使用JDK开发WebService
一.WebService的开发方法 使用java的WebService时可以使用一下两种开发手段 使用jdk开发(1.6及以上版本) 使用CXF框架开发(工作中) 二.使用JDK开发WebServic ...
- Python - 面向对象编程 - 实战(6)
需求 设计一个培训机构管理系统,有总部.分校,有学员.老师.员工,实现具体如下需求: 有多个课程,课程要有定价 有多个班级,班级跟课程有关联 有多个学生,学生报名班级,交这个班级对应的课程的费用 有多 ...
- PreparedStatement预编译的sql执行对象
一.预编译,防sql注入 其中,设置参数值占位符索引从1开始:在由sql 连接对象创建 sql执行对象时候传入参数sql语句,在执行对象在执行方法时候就不用再传入sql语句: 数据库索引一般是从1开始 ...
- 开源自己编写的半人工标注平台PaddleOCRLabel(.NET Winform版本)
大家好, 我是博客园的老用户了,许久不做.NET技术了,从2013年起,开始从事App技术,写过书,在Linux上搭建区块链,用GO写智能合约,使用nodejs搭建过微服务,用python写过爬虫,写 ...
- 分布式搜索引擎Elasticsearch在CentOS7中的安装
1. 概述 随着企业业务量的不断增大,业务数据随之增加,传统的基于关系型数据库的搜索已经不能满足需要. 在关系型数据库中搜索,只能支持简单的关键字搜索,做不到分词和统计的功能,而且当单表数据量到达上百 ...