hdu 6010 路径交(lca + 线段树)

题意:

给出一棵大小为\(n\)的树和\(m\)条路径,求第\(L\)条路径到第\(R\)条路径的交的路径的长度

思路:

本题的关键就是求路径交

假设存在两条路径(a,b),(c,d),那它们的交是怎样的呢

画图看看,分三种情况

  • 1、路径不存在交集
  • 2、(c,d)交(a,b)路径的一边(由lca(a,b)将路径划分成两部分,假设这里交在a的一侧)
  • 3、(c,d)交(a,b)路径的两边

可以看出路径交若存在,这其两端点一定在lca(a,c),lca(a,d),lca(b,c),lca(b,d)中取到,

而且是取深度最大的两点,如果这两点相同,那么显然路径只交于一点了

这里是问边的交集

如果是求的点的交集,还需要判断一下点是否在路径上,可以用距离来判断

然后剩下的来就是简单的线段树求区间的合并了

#include<bits/stdc++.h>
#define LL long long
#define P pair<int,int>
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
#define ls rt<<1
#define rs (rt<<1|1)
using namespace std;
int read(){
int x = 0;
char c = getchar();
while(c < '0' || c > '9') c = getchar();
while(c >= '0' && c <= '9') x = x * 10 + c - 48, c = getchar();
return x;
}
const int N = 5e5 + 10;
int n,m;
int pre[N];
vector<P> G[N];
int dep[N],dis[N];
int fa[N][25];
int dp[N][25];
int ver[2 * N],tot;
int first[N],R[2 * N];
void dfs(int u,int f,int d,int dw){
dep[u] = d;
dis[u] = dw;
fa[u][0] = f;
ver[++tot] = u;
first[u] = tot,R[tot] = d;
// for(int i = 1;i <= 20;i++) fa[u][i] = fa[fa[u][i-1]][i-1];
for(int i = 0;i < G[u].size();i++){
if(G[u][i].first != f) {
dfs(G[u][i].first,u,d + 1,dw + G[u][i].second);
ver[++tot] = u;
R[tot] = d;
}
}
}
void ST(int n){
for(int i = 1;i <= n;i++) dp[i][0] = i;
for(int j = 1;(1<<j)<=n;j++){
for(int i = 1;i + (1<<j) - 1 <= n;i++){
int a= dp[i][j-1],b = dp[i+(1<<(j-1))][j-1];
dp[i][j] = R[a] < R[b]?a:b;
}
}
}
int RMQ(int l,int r){
int k = pre[r - l + 1];
int a = dp[l][k],b = dp[r-(1<<k)+1][k];
return R[a] < R[b]?a:b;
}
int lca(int u,int v){
int x = first[u],y = first[v];
if(x > y) swap(x,y);
int res = RMQ(x,y);
return ver[res];
}/*
int lca(int u,int v){
if(dep[u] < dep[v]) swap(u,v);
int d = dep[u] - dep[v];
for(int i = 20;i >= 0 && u != v;i--) if(d & (1<<i)) u = fa[u][i];
if(u == v) return u;
for(int i = 20;i >= 0;i--) if(fa[u][i] != fa[v][i]) u = fa[u][i],v = fa[v][i];
return fa[u][0];
}*/
int getdis(int u,int v){
return dis[u] + dis[v] - 2 * dis[lca(u,v)];
}
void sw(int &x,int &y){
if(dep[x] < dep[y]) swap(x,y);
}
struct Line{
int u,v,uv;
Line(int u,int v,int uv):u(u),v(v),uv(uv){};
Line(){};
Line operator+(const Line &rhs)const{
if(uv == 0) return rhs;
if(rhs.uv == 0)return *this;
if(uv == -1 || rhs.uv == -1) return Line(0,0,-1);
Line ans;
int lu = lca(u,rhs.u), lv = lca(u,rhs.v);
int ru = lca(v,rhs.u), rv = lca(v,rhs.v);
sw(lu,lv);sw(ru,rv);
if(dep[lu] > dep[ru]){
ans.u = lu;
ans.v = dep[ru] > dep[lv]?ru:lv;
}else {
ans.u = ru;
ans.v = dep[lu] > dep[rv]?lu:rv;
}
if(ans.u == ans.v) return Line(0,0,-1);
ans.uv = lca(ans.u,ans.v);
return ans;
}
}s[N << 2];
void build(int l,int r,int rt){
if(l == r){
s[rt].u = read();
s[rt].v = read();
s[rt].uv = lca(s[rt].u,s[rt].v);
return ;
}
int m = (l + r)>>1;
build(l,m,rt<<1);
build(m+1,r,rt<<1|1);
s[rt] = s[rt<<1] + s[rt<<1|1];
}
Line query(int L,int R,int l,int r,int rt){
if(L <= l && R >= r) return s[rt];
int m = l + r >>1;
Line ans = Line(0,0,0);
if(L <= m) ans = ans + query(L,R,l,m,rt<<1);
if(R > m) ans = ans + query(L,R,m+1,r,rt<<1|1);
return ans;
}
int main(){
for(int i = 1;i < N ;i++) pre[i] = log2(i);
int u,v,w;
while(scanf("%d",&n)==1){
for(int i = 1;i < n;i++){
u = read(),v = read(), w = read();
G[u].push_back(P(v,w));
G[v].push_back(P(u,w));
}
dfs(1,-1,0,0);
ST(2 * n - 1);
int m = read();
build(1,m,1);
int q = read(),L,R;
while(q--){
L = read(),R = read();
Line ans = query(L,R,1,m,1);
if(ans.uv == -1) printf("%d\n",0);
else printf("%d\n",dis[ans.u] + dis[ans.v] - dis[ans.uv] * 2);
}
}
return 0;
}

hdu 6010 路径交的更多相关文章

  1. 2017"百度之星"程序设计大赛 - 初赛(A) [ hdu 6108 小C的倍数问题 ] [ hdu 6109 数据分割 ] [ hdu 6110 路径交 ] [ hdu 6112 今夕何夕 ] [ hdu 6113 度度熊的01世界 ]

    这套题体验极差. PROBLEM 1001 - 小C的倍数问题 题 OvO http://acm.hdu.edu.cn/showproblem.php?pid=6108 (2017"百度之星 ...

  2. hdu 5700区间交(线段树)

    区间交 Time Limit: 8000/4000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Submiss ...

  3. hihocoder[Offer收割]编程练习赛19 D 相交的铁路线(树上路径交)

    傻逼题... 裸的树上路径交 两条树上的路径$[a,b]$和$[c,d]$有交,则有$lca(a,b)$在$[c,d]$上或$lca(c,d)$在$[a,b]$上. 其实只要深度大的$lca$在另一条 ...

  4. 【百度之星初赛A】路径交 LCA+线段树

    [百度之星初赛A]路径交 Problem Description 给定一棵n个点的树,以及m条路径,每次询问第L条到第R条路径的交集部分的长度(如果一条边同时出现在2条路径上,那么它属于路径的交集). ...

  5. hdu6110:路径交

    $n \leq 500000$的树给$m \leq 500000$个路径,$q \leq 500000$个询问每次问一个区间的路径交. 路径交口诀:(前方高能) 判有交,此链有彼祖: 取其交,最深两两 ...

  6. HDU 6116 路径计数

    HDU 6116 路径计数 普通生成函数常用于处理组合问题,指数生成函数常用于处理排列问题. 考虑 对于 $ a $ 个 $ A $ 分为很多堆,这么分的方案数是 $ C_{a-1}^{i-1} $ ...

  7. 【HDU】6110 路径交(2017百度之星) 线段树+RMQ-LCA+树链的交

    [题目]2017"百度之星"程序设计大赛 - 初赛(A) [题意]给定n个点的带边权树,m条编号1~m的路径,Q次询问编号区间[L,R]所有链的交集的长度.n<=500000 ...

  8. HDU 5700 区间交(线段树)

    [题目链接] http://acm.hdu.edu.cn/showproblem.php?pid=5700 [题目大意] 给出一个长度为n的数列和m个区间,现在求k个区间,使得他们的区间交内的数列项和 ...

  9. HDU 5700 区间交 离线线段树

    区间交 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5700 Description 小A有一个含有n个非负整数的数列与m个区间.每个区间可以表示为 ...

随机推荐

  1. 【MYSQL笔记2】复制表,在已有表的基础上设置主键,insert和replace

    之前我自己建立好了一个数据库xscj:表xs是已经定义好的 具体的定义数据类型如下: 为了复制表xs,我们新建一个表名为xstext,使用下列语句进行复制xs,或者说是备份都可以: create ta ...

  2. Docker自学纪实(二)Docker基本操作

    安装docker 以CentOS7为例: 安装:yum -y install docker 启动:systemctl start docker 设置开机自启:systemctl enable dock ...

  3. LNMP+HAProxy+Keepalived负载均衡 - 基础服务准备

    日志服务 修改日志服务配置并重启日志服务: ``` vim /etc/rsyslog.conf ``` 编辑系统日志配置,指定包含的配置文件路径和规则: ``` $IncludeConfig /etc ...

  4. java中substring()、charAt()、indexOf() (2013-05-05-bd 写的日志迁移

    substring 1. public String substring(int beginIndex)     返回一个新的字符串,它是此字符串的一个子字符串, 该子字符串始于指定索引处的字符,一直 ...

  5. python创建字典

    创建: {x:x**2 for x in (2,4,6)} dict(xjm=110,lxh=119,pzq=120) dict([('a',1),('b',2),('c',3)])

  6. [BZOJ2527] [Poi2011]Meteors(整体二分)

    对于单个国家,可以对答案进行二分,每次找出此时的陨石数量,如果大于需要的那么答案就在[l,mid],否则就在[mid+1,r]里面 而对于很多国家,也可以进行二分,solve(l,r,L,R)表示询问 ...

  7. Android 自定义WebView 实现可以加载缓存数据

    1.自定义WebView说明 1.1.这个WebView可以加载缓存的数据.(需要后端配合,将html转换成一个字符串,主要是图片要用特殊格式) 1.2.注入了图片链接,为了方便点击webView中的 ...

  8. 6,Flask 中内置的 Session

    Flask中的Session非常的奇怪,他会将你的SessionID存放在客户端的Cookie中,使用起来也非常的奇怪 1. Flask 中 session 是需要 secret_key 的 from ...

  9. 3 Mongodb数据查询1

    1.基本查询 方法find():查询 db.集合名称.find({条件文档}) 方法findOne():查询,只返回第一个 db.集合名称.findOne({条件文档}) 方法pretty():将结果 ...

  10. 2037: [Sdoi2008]Sue的小球

    2037: [Sdoi2008]Sue的小球 链接 题解 论文 代码 #include<cstdio> #include<algorithm> #include<cstr ...