题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6162

题意:给出一棵树的链接方法,每个点都有一个数字,询问U-》V节点经过所有路径中l < = x < = r的数字和

解法:主席树维护区间和,树剖查询,复杂度nloglog。

代码:

#include <bits/stdc++.h>
using namespace std;
const int maxn = 1e5+5;
const int maxm = 40*maxn;
typedef long long LL;
int T[maxn];
int lson[maxm],rson[maxm];
LL sum[maxm];
int clk,M;
int newnode(){
clk++;
lson[clk]=rson[clk]=sum[clk]=0;
return clk;
}
void update(int &now, int pre, int L, int R, int pos, int val)
{
now = newnode();
lson[now] = lson[pre];
rson[now] = rson[pre];
sum[now] = sum[pre] + val;
if(L!=R){
int mid=(L+R)>>1;
if(pos<=mid) update(lson[now],lson[pre],L,mid,pos,val);
else update(rson[now],rson[pre],mid+1,R,pos,val);
}
}
LL query(int rt, int L, int R, int l, int r){
if(l<=L&&R<=r){
return sum[rt];
}
else{
int mid = (L+R)>>1;
LL ret = 0;
if(l <= mid) ret += query(lson[rt], L, mid, l, r);
if(mid < r) ret += query(rson[rt], mid+1, R, l, r);
return ret;
}
}
struct edge{
int to,next;
}E[maxn*2];
int head[maxn],edgecnt,tim;
int siz[maxn],top[maxn],son[maxn],dep[maxn];
int fa[maxn],tid[maxn],Rank[maxn],val[maxn];
void init(){
edgecnt=tim=0;
memset(head,-1,sizeof(head));
memset(son,-1,sizeof(son));
clk=M=0;
lson[clk]=rson[clk]=sum[clk]=0;
}
void add(int u, int v){
E[edgecnt].to=v,E[edgecnt].next=head[u],head[u]=edgecnt++;
}
void dfs1(int u, int pre, int d){
dep[u]=d;
fa[u]=pre;
siz[u]=1;
for(int i=head[u];~i;i=E[i].next){
int v=E[i].to;
if(v!=pre){
dfs1(v,u,d+1);
siz[u]+=siz[v];
if(son[u]==-1||siz[v]>siz[son[u]]) son[u]=v;
}
}
}
void dfs2(int u, int tp){
top[u]=tp;
tid[u]=++tim;
Rank[tid[u]]=u;
if(son[u]==-1) return;
dfs2(son[u],tp);
for(int i=head[u];~i;i=E[i].next){
int v=E[i].to;
if(v!=son[u]&&v!=fa[u]){
dfs2(v,v);
}
}
}
int LCA(int u, int v)
{
int ret;
while(true)
{
if(top[u] == top[v])
{
ret = dep[u] < dep[v] ? u : v;
break;
}
else if(dep[top[u]] > dep[top[v]])
u = fa[top[u]];
else v = fa[top[v]];
}
return ret;
}
int t[maxn];
LL query(int u, int v, int a, int b)
{
a = lower_bound(t + 1, t + 1 + M, a) - t - 1;
b = upper_bound(t + 1, t + 1 + M, b) - t - 1;
if(b == 0)
return 0;
LL ret = 0;
while(top[u] != top[v])
{
if(dep[top[u]] < dep[top[v]])
swap(u, v);
ret += query(T[tid[u]], 1, M, 1, b);
ret -= query(T[tid[top[u]] - 1], 1, M, 1, b);
if(a)
{
ret -= query(T[tid[u]], 1, M, 1, a);
ret += query(T[tid[top[u]] - 1], 1, M, 1, a);
}
u = fa[top[u]];
}
if(dep[u] < dep[v])
swap(u, v);
ret += query(T[tid[u]], 1, M, 1, b);
ret -= query(T[tid[v] - 1], 1, M, 1, b);
if(a)
{
ret -= query(T[tid[u]], 1, M, 1, a);
ret += query(T[tid[v] - 1], 1, M, 1, a);
}
return ret;
}
int main()
{
int n,m;
while(~scanf("%d %d",&n,&m))
{
init();
for(int i=1; i<=n; i++){
scanf("%d", &val[i]);
t[i]=val[i];
}
for(int i=1; i<n; i++){
int u,v;
scanf("%d %d", &u,&v);
add(u, v);
add(v, u);
}
dfs1(1,0,0);
dfs2(1,1);
sort(t+1,t+n+1);
M = unique(t+1, t+n+1)-t-1;
for(int i=1; i<=n; i++) val[i] = lower_bound(t+1, t+1+M, val[i])-t;
for(int i=1; i<=n; i++){
T[i] = T[i-1];
update(T[i],T[i],1,M,val[Rank[i]],t[val[Rank[i]]]);
}
for(int i=1; i<=m; i++){
int u,v,x,y;
scanf("%d %d %d %d", &u,&v,&x,&y);
LL ret = query(u,v,x,y);
printf("%lld", ret);
if(i == m){
printf("\n");
}
else{
printf(" ");
}
}
}
return 0;
}

2017多校第9场 HDU 6162 Ch’s gift 树剖加主席树的更多相关文章

  1. 2017多校第9场 HDU 6161 Big binary tree 思维,类似字典树

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6161 题意: 题目是给一棵完全二叉树,从上到下从左到右给每个节点标号,每个点有权值,初始权值为其标号, ...

  2. HDU 6162 - Ch’s gift | 2017 ZJUT Multi-University Training 9

    /* HDU 6162 - Ch’s gift [ LCA,线段树 ] | 2017 ZJUT Multi-University Training 9 题意: N节点的树,Q组询问 每次询问s,t两节 ...

  3. 2017 Multi-University Training Contest - Team 9 1002&&HDU 6162 Ch’s gift【树链部分+线段树】

    Ch’s gift Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total S ...

  4. 2017多校第9场 HDU 6170 Two strings DP

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6170 题意:给了2个字符串,其中第2个字符串包含.和*两种特别字符,问第二个字符串能否和第一个匹配. ...

  5. 2017多校第9场 HDU 6169 Senior PanⅡ 数论,DP,爆搜

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6169 题意:给了区间L,R,求[L,R]区间所有满足其最小质数因子为k的数的和. 解法: 我看了这篇b ...

  6. 2017多校第10场 HDU 6181 Two Paths 次短路

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6181 题意:给一个图,求出次短路. 解法:我之前的模板不能解决这种图,就是最短路和次短路相等的情况,证 ...

  7. 2017多校第10场 HDU 6180 Schedule 贪心,multiset

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6180 题意:给了一些任务的开始时间和终止时间,现在让我们安排k台及机器,让这些任务在k太机器上最小,并 ...

  8. 2017多校第10场 HDU 6178 Monkeys 贪心,或者DP

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6178 题意:给出一棵有n个节点的树,现在需要你把k只猴子放在节点上,每个节点最多放一只猴子,且要求每只 ...

  9. 2017多校第10场 HDU 6171 Admiral 双向BFS或者A*搜索

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6171 题意: 给你一个高度为6的塔形数组,你每次只能将0与他上下相邻的某个数交换,问最少交换多少次可以 ...

随机推荐

  1. bzoj1835[ZJOI2010]基站选址

    主席树+决策单调,重写一遍比之前短多了……题解:http://www.cnblogs.com/liu-runda/p/6051422.html #include<cstdio> #incl ...

  2. bzoj2165: 大楼 (矩阵快速幂)

    //========================== 蒟蒻Macaulish:http://www.cnblogs.com/Macaulish/  转载要声明! //=============== ...

  3. [HNOI2009]有趣的数列 卡特兰数

    题面:[HNOI2009]有趣的数列 题解: 观察到题目其实就是要求从长为2n的序列中选n个放在集合a,剩下的放在集合b,使得集合a和集合b中可以一一对应的使a中的元素小于b. 2种想法(实质上是一样 ...

  4. bzoj1042: [HAOI2008]硬币购物(DP+容斥)

    1600+人过的题排#32还不错嘿嘿 浴谷夏令营讲过的题,居然1A了 预处理出f[i]表示购买价值为i的东西的方案数 然后每次询问进行一次容斥,答案为总方案数-第一种硬币超限方案-第二种超限方案-第三 ...

  5. [APIO2017]商旅

    link 这题卡我精度,调了一晚上才调对,因为没有想到图还可以不连通 其实可以预处理出好多东西,距离($dis(u,v)$),买卖物品(从$u$到$v$买卖物品的最大利润,例($max{S_{u,i} ...

  6. C++语言中数组指针和指针数组彻底分析

    #################################                              ##       基本知识               ##        ...

  7. UIView的autoresizingMask属性研究

    在 UIView 中有一个autoresizingMask的属性,它对应的是一个枚举的值(如下),属性的意思就是自动调整子控件与父控件中间的位置,宽高. 1 2 3 4 5 6 7 8 9 enum  ...

  8. 为什么使用centos部署服务器

    这个是实验室同学面试的时候,面试官问的一个问题? 为什么选择centos系统,为什么centos系统用的比较多呢? 首先我们说下redhat红帽公司,它是全球最大的linux服务提供商,它的服务是最好 ...

  9. UVA11426

    链接:http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&category=26&page ...

  10. DEBUG宏

    4.8.6.运算中的临时匿名变量4.8.6.1.C语言和汇编的区别(汇编完全对应机器操作,C对应逻辑操作)(1)C语言叫高级语言,汇编语言叫低级语言.(2)低级语言的意思是汇编语言和机器操作相对应,汇 ...