题目传送门

先建出来点分树,以每个点为根开线段树,维护点分子树内编号为$[l,r]$的儿子到根的距离最小值

每次查询$x$开始,沿着点分树向上跑,在每个点的线段树的$[l,r]$区间里都查一遍取$min$即可

因为题目让我们求最小值,所以出现重复经过同一条路径的情况并不会让答案变坏

如果让我们求最大值呢?似乎可以用主席树搞搞?(仅仅是口胡有空再想)

 #include <cmath>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define N1 100005
#define M1 505
#define ll long long
#define uint unsigned int
#define ush unsigned short
using namespace std;
const int inf=0x3f3f3f3f; template <typename _T> void read(_T &ret)
{
ret=; _T fh=; char c=getchar();
while(c<''||c>''){ if(c=='-') fh=-; c=getchar(); }
while(c>=''&&c<=''){ ret=ret*+c-''; c=getchar(); }
ret=ret*fh;
} struct Edge{
int to[N1*],nxt[N1*],val[N1*],head[N1],cte;
void ae(int u,int v,int w)
{ cte++; to[cte]=v; nxt[cte]=head[u]; val[cte]=w; head[u]=cte; }
}e; int lg[N1*],de;
namespace Tree{
int dis[N1],dep[N1],fa[N1],st[N1],ff[N1*][],cur; //ord[N1*2],
void dfs1(int x)
{
int j,v; st[x]=++cur; ff[cur][]=x;
for(j=e.head[x];j;j=e.nxt[j])
{
v=e.to[j]; if(v==fa[x]) continue;
fa[v]=x; dis[v]=dis[x]+e.val[j]; dep[v]=dep[x]+;
dfs1(v); ff[++cur][]=x;
}
}
void init()
{
dep[]=; dfs1();
int i,j;
for(i=,lg[]=;i<=cur;i++) lg[i]=lg[i>>]+;
for(j=;j<=lg[cur];j++)
for(i=;i+(<<j)-<=cur;i++)
ff[i][j]=dep[ff[i][j-]] < dep[ff[i+(<<(j-))][j-]] ? ff[i][j-] : ff[i+(<<(j-))][j-];
}
inline int Dis(int x,int y)
{
int i=st[x],j=st[y],F; if(i>j) swap(i,j);
F=dep[ff[i][lg[j-i+]]] < dep[ff[j-(<<lg[j-i+])+][lg[j-i+]]] ? ff[i][lg[j-i+]] : ff[j-(<<lg[j-i+])+][lg[j-i+]];
return dis[x]+dis[y]-*dis[F];
}
}; struct SEG{
struct node{ int ls,rs,mi; };
node a[N1*]; int root[N1],tot;
inline void pushup(int rt)
{
a[rt].mi=inf;
if(a[rt].ls) a[rt].mi=min(a[rt].mi,a[a[rt].ls].mi);
if(a[rt].rs) a[rt].mi=min(a[rt].mi,a[a[rt].rs].mi);
}
void upd(int x,int l,int r,int &rt,int w)
{
if(!rt) rt=++tot;
if(l==r){ a[rt].mi=w; return; }
int mid=(l+r)>>;
if(x<=mid) upd(x,l,mid,a[rt].ls,w);
else upd(x,mid+,r,a[rt].rs,w);
pushup(rt);
}
int qmi(int L,int R,int l,int r,int rt)
{
if(!rt) return inf;
if(L<=l&&r<=R) return a[rt].mi;
int mid=(l+r)>>; int ans=inf;
if(L<=mid) ans=min(ans,qmi(L,R,l,mid,a[rt].ls));
if(R>mid) ans=min(ans,qmi(L,R,mid+,r,a[rt].rs));
return ans;
}
}s; using Tree::Dis; int n;
int sz[N1],ms[N1],use[N1],pfa[N1],SZ,G; void gra(int x,int dad,int now)
{
int j,v; sz[x]=; ms[x]=;
if(now) s.upd(x,,n,s.root[now],Dis(x,now));
for(j=e.head[x];j;j=e.nxt[j])
{
v=e.to[j]; if(v==dad || use[v]) continue;
gra(v,x,now);
sz[x]+=sz[v]; ms[x]=max(ms[x],sz[v]);
}
ms[x]=max(ms[x],SZ-sz[x]);
if(ms[x]<ms[G]) G=x;
} void main_dfs(int x)
{
int j,v; use[x]=;
s.upd(x,,n,s.root[x],);
for(j=e.head[x];j;j=e.nxt[j])
{
v=e.to[j]; if(use[v]) continue;
SZ=sz[v]; G=; gra(v,x,x); pfa[G]=x;
main_dfs(G);
}
} int main()
{
scanf("%d",&n);
int i,j,x,y,w,Q,q,l,r,D,ans;
for(i=;i<n;i++)
{
read(x), read(y), read(w);
e.ae(x,y,w); e.ae(y,x,w);
}
Tree::init();
ms[]=n; SZ=n; gra(,,); main_dfs(G);
scanf("%d",&Q);
for(q=;q<=Q;q++)
{
read(l), read(r), read(x); ans=inf;
for(y=x;y;y=pfa[y])
{
D=Dis(x,y);
ans=min(ans,D+s.qmi(l,r,,n,s.root[y]));
}
printf("%d\r\n",ans);
}
return ;
}

LOJ 6145 Easy (动态点分治+线段树)的更多相关文章

  1. 【loj6145】「2017 山东三轮集训 Day7」Easy 动态点分治+线段树

    题目描述 给你一棵 $n$ 个点的树,边有边权.$m$ 次询问,每次给出 $l$ .$r$ .$x$ ,求 $\text{Min}_{i=l}^r\text{dis}(i,x)$ . $n,m\le ...

  2. 【bzoj4372】烁烁的游戏 动态点分治+线段树

    题目描述 给一颗n个节点的树,边权均为1,初始点权均为0,m次操作:Q x:询问x的点权.M x d w:将树上与节点x距离不超过d的节点的点权均加上w. 输入 第一行两个正整数:n,m接下来的n-1 ...

  3. 【bzoj3730】震波 动态点分治+线段树

    题目描述 在一片土地上有N个城市,通过N-1条无向边互相连接,形成一棵树的结构,相邻两个城市的距离为1,其中第i个城市的价值为value[i].不幸的是,这片土地常常发生地震,并且随着时代的发展,城市 ...

  4. BZOJ4372烁烁的游戏——动态点分治+线段树(点分树套线段树)

    题目描述 背景:烁烁很喜欢爬树,这吓坏了树上的皮皮鼠.题意:给定一颗n个节点的树,边权均为1,初始树上没有皮皮鼠.烁烁他每次会跳到一个节点u,把周围与他距离不超过d的节点各吸引出w只皮皮鼠.皮皮鼠会被 ...

  5. [bzoj4372] 烁烁的游戏 [动态点分治+线段树+容斥原理]

    题面 传送门 思路 观察一下题目,要求的是修改"距离点$u$的距离一定的点权值",那这个就不能用传统的dfs序类算法+线段树维护,因为涉及到向父亲回溯的问题 看到和树上距离相关的东 ...

  6. 2019ICPC上海网络赛 A Lightning Routing I 点分树(动态点分治)+线段树

    题意 给一颗带边权的树,有两种操作 \(C~e_i~w_i\),将第\(e_i\)条边的边权改为\(w_i\). \(Q~v_i\),询问距\(v_i\)点最远的点的距离. 分析 官方题解做法:动态维 ...

  7. BZOJ 4372/3370 烁烁的游戏/震波 (动态点分治+线段树)

    烁烁的游戏 题目大意: 给你一棵$n$个节点的树,有$m$次操作,询问某个节点的权值,或者将与某个点$x$距离不超过$d$的所有节点的权值都增加$w$ 动态点分裸题 每个节点开一棵权值线段树 对于修改 ...

  8. BZOJ3730震波——动态点分治+线段树(点分树套线段树)

    题目描述 在一片土地上有N个城市,通过N-1条无向边互相连接,形成一棵树的结构,相邻两个城市的距离为1,其中第i个城市的价值为value[i].不幸的是,这片土地常常发生地震,并且随着时代的发展,城市 ...

  9. BZOJ4372 烁烁的游戏(动态点分治+线段树)

    建出点分树,每个节点维护其作为点分树上lca对子树内点的贡献,线段树维护即可,同时另开一个线段树以减掉父亲重复的贡献. #include<iostream> #include<cst ...

随机推荐

  1. HDU 5289 Assignment (二分+区间最值)

    [题目链接]click here~~ [题目大意]: 给出一个数列,问当中存在多少连续子序列,子序列的最大值-最小值<k [思路]:枚举数列左端点.然后二分枚举右端点,用ST算法求区间最值.(或 ...

  2. 零基础如何学习Java和web前端

    今天说一下零基础到底能不能学习Java,为什么有的人说学不了呢,那么接下来我为大家揭晓,零基础到底适合不适合学习Java. 零基础学习Java的途径第一个就是看视频,然后就是看书,或者在线下报个培训班 ...

  3. Java中static方法

    今天学习到了并且应用到了java中的静态方法,并且了解到它的好处与缺点. ● 生命周期(Lifecycle): 静态方法(Static Method)与静态成员变量一样,属于类本身,在类装载的时候被装 ...

  4. AOP注解不起作用的debug结果

    经过2天的调试,我发现AOP注解配置不起作用居然是表达式的错误导致的 在xml文件中配置的base-package有关,初步认为@PointCut只能使用base-package..*(..)这样的方 ...

  5. Scala-基础-数据类型

    import junit.framework.TestCase import org.junit.Test import scala.runtime.RichByte //数据类型 class Dem ...

  6. 元素属性的添加删除(原生js)

    添加属性 odiv.setAttribute("title","hello div!"); odiv.setAttribute("class" ...

  7. 最容易理解的CSS的position教程——十步图解CSS的position

    CSS的positon,我想做为一个Web制作者来说都有碰到过,但至于对其是否真正的了解呢?那我就不也说了,至少我自己并不非常的了解其内核的运行.今天在Learn CSS Positioning in ...

  8. Gradle的属性Property设置与调用

    Gradle在默认情况下已经为Project定义了很多Property: project:Project本身 name:Project的名字 path:Project的绝对路径 description ...

  9. Android项目实战_手机安全卫士拦截骚扰

    ###1.骚扰拦截需求分析1.界面1.1 黑名单列表界面1.2 添加黑名单界面2.功能2.1 黑名单的添加.删除2.2 拦截电话2.3 拦截短信 ###2.黑名单数据库的创建1.分析需要的字段id 主 ...

  10. win32窗口映射(部分)

    先理解一下“窗口”与“视区”的概念.“窗口”是逻辑坐标下的矩形区域,“视区”是设备坐标系下的区域.根据“窗口”和“视区”的大小可以确定x方向和y方向的比例因子. 例子如下: VOID OnPaint( ...