[51nod 1830] 路径交
问题描述
给定一棵n个点的树,以及m条路径,每次询问第L条到第R条路径的交集部分的长度(如果一条边同时出现在2条路径上,那么它属于路径的交集)。
输入格式
第一行一个数n(n<=500,000)
接下来n-1行,每行三个数x,y,z,表示一条从x到y并且长度为z的边
第n+1行一个数m(m<=500,000)
接下来m行,每行两个数u,v,表示一条从u到v的路径
接下来一行一个数Q,表示询问次数(Q<=500,000)
接下来Q行,每行两个数L和R
输出格式
Q行,每行一个数表示答案
样例输入
4
1 2 5
2 3 2
1 4 3
2
1 2
3 4
1
1 2
样例输出
5
解析
按照常规思想,这也许可以通过一些树上路径相关的数据结构来完成。但是实现起来有很大的困难。
看到询问的方式,与线段树区间询问的类型很相似。不妨也用类似的方法,用线段树维护路径的交集。具体维护起来可以通过维护交集路径的两个端点,加上一些分类讨论完成。为了保证复杂度是\(O(nlogn)\)的,需要用ST表实现最近公共祖先。
代码
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cmath>
#define N 500002
using namespace std;
const int inf=1<<30;
struct path{
int u,v,dis;
}a[N],t[N*4];
int head[N],ver[N*2],nxt[N*2],edge[N*2],l;
int n,m,q,i,st[N][30],dep[N],dis[N],in[N],s[N],top;
int read()
{
char c=getchar();
int w=0;
while(c<'0'||c>'9') c=getchar();
while(c<='9'&&c>='0'){
w=w*10+c-'0';
c=getchar();
}
return w;
}
void insert(int x,int y,int z)
{
l++;
ver[l]=y;
edge[l]=z;
nxt[l]=head[x];
head[x]=l;
}
void dfs(int x,int pre)
{
s[++top]=x;
dep[top]=dep[in[pre]]+1;
in[x]=top;
for(int i=head[x];i;i=nxt[i]){
int y=ver[i];
if(y!=pre){
dis[y]=dis[x]+edge[i];
dfs(y,x);
s[++top]=x,dep[top]=dep[in[x]];
}
}
}
int get(int x,int y)
{
if(dep[x]>dep[y]) return y;
return x;
}
void init()
{
dfs(1,0);
for(int i=1;i<=top;i++) st[i][0]=i;
for(int j=0;(1<<(j+1))<=top;j++){
for(int i=1;i+(1<<(j+1))-1<=top;i++) st[i][j+1]=get(st[i][j],st[i+(1<<j)][j]);
}
}
int LCA(int u,int v)
{
int l=in[u],r=in[v];
if(l>r) swap(l,r);
int k=log2(1.0*(r-l+1));
return s[get(st[l][k],st[r-(1<<k)+1][k])];
}
int dist(int u,int v)
{
return dis[u]+dis[v]-2*dis[LCA(u,v)];
}
int my_comp(const int &x,const int &y)
{
return dep[in[x]]>dep[in[y]];
}
path update(path x,path y)
{
if(x.dis==inf) return y;
if(y.dis==inf) return x;
if(x.u==y.u&&x.v==y.v) return x;
if(x.dis==0||y.dis==0) return (path){0,0,0};
int a[4];
a[0]=LCA(x.u,y.u);
a[1]=LCA(x.u,y.v);
a[2]=LCA(x.v,y.u);
a[3]=LCA(x.v,y.v);
sort(a,a+4,my_comp);
int p1=a[0],p2=a[1];
if(p1==p2&&((dep[in[p1]]<dep[in[LCA(x.u,x.v)]])||(dep[in[p2]]<dep[in[LCA(y.u,y.v)]]))) return (path){0,0,0};
return (path){p1,p2,dist(p1,p2)};
}
void build(int p,int l,int r)
{
if(l==r){
t[p].u=a[l].u,t[p].v=a[l].v;
t[p].dis=dist(t[p].u,t[p].v);
return;
}
int mid=(l+r)/2;
build(p*2,l,mid);
build(p*2+1,mid+1,r);
t[p]=update(t[p*2],t[p*2+1]);
}
path ask(int p,int l,int r,int ql,int qr)
{
if(ql<=l&&r<=qr) return t[p];
int mid=(l+r)/2;
path ansl=(path){0,0,inf},ansr=(path){0,0,inf};
if(ql<=mid) ansl=ask(p*2,l,mid,ql,qr);
if(qr>mid) ansr=ask(p*2+1,mid+1,r,ql,qr);
return update(ansl,ansr);
}
int main()
{
n=read();
for(i=1;i<n;i++){
int u=read(),v=read(),w=read();
insert(u,v,w);
insert(v,u,w);
}
init();
m=read();
for(i=1;i<=m;i++) a[i].u=read(),a[i].v=read();
build(1,1,m);
q=read();
for(i=1;i<=q;i++){
int l=read(),r=read();
path ans=ask(1,1,m,l,r);
printf("%d\n",ans.dis);
}
return 0;
}
[51nod 1830] 路径交的更多相关文章
- hihocoder[Offer收割]编程练习赛19 D 相交的铁路线(树上路径交)
傻逼题... 裸的树上路径交 两条树上的路径$[a,b]$和$[c,d]$有交,则有$lca(a,b)$在$[c,d]$上或$lca(c,d)$在$[a,b]$上. 其实只要深度大的$lca$在另一条 ...
- hdu 6010 路径交
hdu 6010 路径交(lca + 线段树) 题意: 给出一棵大小为\(n\)的树和\(m\)条路径,求第\(L\)条路径到第\(R\)条路径的交的路径的长度 思路: 本题的关键就是求路径交 假设存 ...
- 【百度之星初赛A】路径交 LCA+线段树
[百度之星初赛A]路径交 Problem Description 给定一棵n个点的树,以及m条路径,每次询问第L条到第R条路径的交集部分的长度(如果一条边同时出现在2条路径上,那么它属于路径的交集). ...
- hdu6110:路径交
$n \leq 500000$的树给$m \leq 500000$个路径,$q \leq 500000$个询问每次问一个区间的路径交. 路径交口诀:(前方高能) 判有交,此链有彼祖: 取其交,最深两两 ...
- 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"百度之星 ...
- 51nod 1443 路径和树(最短路)
题目链接:https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1443 1443 路径和树 题目来源: CodeForces ...
- 【HDU】6110 路径交(2017百度之星) 线段树+RMQ-LCA+树链的交
[题目]2017"百度之星"程序设计大赛 - 初赛(A) [题意]给定n个点的带边权树,m条编号1~m的路径,Q次询问编号区间[L,R]所有链的交集的长度.n<=500000 ...
- 51Nod 1967 路径定向 —— 欧拉回路
题目:http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1967 显然是欧拉回路问题,度数为奇数的点之间连边,跑欧拉回路就可以 ...
- 51nod 1443 路径和树(最短路树)
题目链接:路径和树 题意:给定无向带权连通图,求从u开始边权和最小的最短路树,输出最小边权和. 题解:构造出最短路树,把存留下来的边权全部加起来.(跑dijkstra的时候松弛加上$ < $变成 ...
随机推荐
- centos7 忘记root密码,如何进入单用户模式。
init方法 1.centos7的grub2界面会有两个入口,正常系统入口和救援模式: 2.修改grub2引导 在正常系统入口上按下"e",会进入edit模式,搜寻ro那一行,以l ...
- XSS的简单过滤和绕过
XSS的简单过滤和绕过 程序猿用一些函数将构成xss代码的一些关键字符给过滤了.但是,道高一尺魔高一丈,虽然过滤了,还是可以尝试进行过滤绕过,以达到XSS攻击的目的. 最简单的是输入<scrip ...
- TIOBE11月份编程语言排行榜:C非常接近Java,分析下中美的就业情况
TIOBE公布11月份编程语言排行榜:C非常接近Java Swift挤进前10,分析下中美的就业情况. 我们先看看他们官方对数据的解读 本月TIOBE指数前20位出现了一些有趣的变动.首先,C语言现在 ...
- websocket服务器推送 (node+express+vue+socket)
简介: 此项目需要懂一点node.express 功能: 1.前端用户登录,查看服务端推送的消息,用户只能在一个地方登录,也就是单点登录 2.服务端首先登录,上传需要推送的信息文本,后台读取文本后,存 ...
- Elasticsearch-数据的存储、搜索(干货)
ES-深入功能ES中数据是如何组织的?逻辑设计:用于索引和搜索的基本单位是文档,可以将其认为是关系数据库里的一行.文档以类型来分组,类型包含若干文档,类似表格包含若干行.最终,一个或多个类型存在于同一 ...
- 另一种分页器 不依赖Paginator模块的方法
""" 分页组件 """ class Pagination(object): def __init__(self, current_page ...
- Go语言代码结构与语法基础(二)
任何一门语言,都是从打印 hello world 开始的. 最简单的go代码: package main // 声明 main 包,表明当前是一个可执行程序 import "fmt" ...
- 03: redis高级
1.1 布隆过滤器 1.布隆过滤器是什么?(判断某个key一定不存在) 1. 本质上布隆过滤器是一种数据结构,比较巧妙的概率型数据结构 2. 特点是高效地插入和查询,可以用来告诉你 “某样东西一定不存 ...
- 06.AutoMapper 之内联映射(Inline Mapping)
https://www.jianshu.com/p/623655d7cb34 内联映射(Inline Mapping) AutoMapper在 6.2 以上版本将动态创建类型映射. 当第一次调用Map ...
- wex5 如何写后台BAAS
Data.java: 在class中链接数据源: 配置的numsql数据源 private static final String DATASOURCE_NUMYSQL = "numysql ...