HDU 6547 Tree 树链剖分+线段树区间开根区间求和

http://acm.hdu.edu.cn/showproblem.php?pid=6547
wls 有三棵树,树上每个节点都有一个值 ai
,现在有 2 种操作:
1. 将一条链上的所有节点的值开根号向下取整;
2. 求一条链上值的和;
链的定义是两点之间的最短路。
Input
第一行两个数 n, q 分别代表树上点的数量和操作数量。
第二行 n 个整数,第 i 个数代表第 i 个点的值 ai。
接下来 n − 1 行, 每行两个整数 u, v 代表 u,v 之间有一条边。数据保证点两两联通。
接下来 q 行,每行有个整数 op, u, v,op = 0 表示将 u, v 这条链上所有的点的值开根号向下取整,op = 1表示询问 u,v 这条链上的值的和。
1 ≤ n, q ≤ 100, 000
0 ≤ ai
≤ 1, 000, 000, 000
Output
对于每一组 op
= 2 的询问,输出一行一个值表示答案。
Sample Input

4 4
2 3 4 5
1 2
2 3
2 4
0 3 4
0 1 3
1 2 3
1 1 4

Sample Output

2
4

思路:模板题。线段树区间开根请戳这里

#include<iostream>
#include<cstdio>
#include<cstring>
#include<vector>
#include<cmath>
using namespace std;
typedef long long ll;
const int maxn=1e5+5;

struct node
{
    int l,r;
    ll MAX,MIN,sum,lazy;
}tree[maxn<<2]; //线段树

int siz[maxn];//子树大小
int son[maxn];//重儿子
int fa[maxn];//父节点
int deep[maxn];//深度
int top[maxn];//所在链链顶
int pos[maxn];//dfs序编号
ll v[maxn];//映射到序列后的值 线段树依照此序列建树
ll a[maxn];//树上各节点的值
vector<int> vec[maxn];//存边
int n,m,tot=0;

void dfs1(int u,int f)//处理出重儿子 深度 父亲 子树大小等信息
{
    siz[u]=1;
    son[u]=0;
    fa[u]=f;
    deep[u]=deep[f]+1;
    int to;
    for(int i=0;i<vec[u].size();i++)
    {
        to=vec[u][i];
        if(to!=f)
        {
            dfs1(to,u);
            siz[u]+=siz[to];
            if(siz[son[u]]<siz[to])
                son[u]=to;
        }
    }
}

void dfs2(int u,int f,int k)//处理出 链顶 dfs序 映射后的值 等信息
{
    top[u]=k;
    pos[u]=++tot;
    v[tot]=a[u];
    if(son[u])
        dfs2(son[u],u,k);
    int to;
    for(int i=0;i<vec[u].size();i++)
    {
        to=vec[u][i];
        if(to!=f&&to!=son[u])
            dfs2(to,u,to);
    }
}

inline void down(int i)
{
    int l=i<<1,r=i<<1|1;
    tree[l].lazy+=tree[i].lazy;
    tree[r].lazy+=tree[i].lazy;
    tree[l].sum+=(tree[l].r-tree[l].l+1)*tree[i].lazy;
    tree[r].sum+=(tree[r].r-tree[r].l+1)*tree[i].lazy;
    tree[l].MAX+=tree[i].lazy;
    tree[l].MIN+=tree[i].lazy;
    tree[r].MAX+=tree[i].lazy;
    tree[r].MIN+=tree[i].lazy;
    tree[i].lazy=0;
}

inline void up(int i)
{
    int l=i<<1,r=i<<1|1;
    tree[i].sum=tree[l].sum+tree[r].sum;
    tree[i].MAX=max(tree[l].MAX,tree[r].MAX);
    tree[i].MIN=min(tree[l].MIN,tree[r].MIN);
}

void build(int i,int l,int r)
{
    tree[i].l=l,tree[i].r=r;
    tree[i].lazy=0;
    if(l==r)
    {
        tree[i].MAX=tree[i].MIN=tree[i].sum=v[l];
        return ;
    }
    int mid=(l+r)>>1;
    build(i<<1,l,mid);
    build(i<<1|1,mid+1,r);
    up(i);
}

ll querysum(int i,int l,int r)
{
    if(tree[i].l==l&&tree[i].r==r)
        return tree[i].sum;
    if(tree[i].lazy)
        down(i);
    int mid=(tree[i].l+tree[i].r)>>1;
    if(r<=mid)
        return querysum(i<<1,l,r);
    else if(l>mid)
        return querysum(i<<1|1,l,r);
    else
        return querysum(i<<1,l,mid)+querysum(i<<1|1,mid+1,r);
}

void update(int i,int l,int r)
{
    if(tree[i].l==l&&tree[i].r==r)
    {
        ll a=sqrt(tree[i].MAX);
        ll b=sqrt(tree[i].MIN);
        ll v=a-tree[i].MAX;
        int flag=0;
        if(tree[i].MAX==tree[i].MIN||tree[i].MAX-a==tree[i].MIN-b)
            flag=1;
        if(flag)
        {
            tree[i].sum+=(r-l+1)*v;
            tree[i].MAX+=v;
            tree[i].MIN+=v;
            tree[i].lazy+=v;
            return ;
        }
    }
    if(tree[i].lazy)
        down(i);
    int mid=(tree[i].l+tree[i].r)>>1;
    if(r<=mid)
        update(i<<1,l,r);
    else if(l>mid)
        update(i<<1|1,l,r);
    else
        update(i<<1,l,mid),
        update(i<<1|1,mid+1,r);
    up(i);
}

ll calsum(int u,int v)//计算路径和
{
    ll ans=0;
    while(top[u]!=top[v])//不在同一条链上 每次链顶深度较大的点往上爬
    {
        if(deep[top[u]]<deep[top[v]])
            swap(u,v);
        ans+=querysum(1,pos[top[u]],pos[u]);
        u=fa[top[u]];//进入新的链
    }
    if(deep[u]>deep[v])//进入同一条链上时再求一遍区间和 深度小的点在前面
        swap(u,v);
    ans+=querysum(1,pos[u],pos[v]);
    return ans;
}

void updatework(int u,int v)
{
    while(top[u]!=top[v])
    {
        if(deep[top[u]]<deep[top[v]])
            swap(u,v);
        update(1,pos[top[u]],pos[u]);
        u=fa[top[u]];
    }
    if(deep[u]>deep[v])
        swap(u,v);
    update(1,pos[u],pos[v]);
}

inline void prework()
{
    scanf("%d%d",&n,&m);
    int u,v;
    for(int i=1;i<=n;i++)
        scanf("%lld",&a[i]);
    for(int i=1;i<n;i++)
    {
        scanf("%d %d",&u,&v);
        vec[u].push_back(v);
        vec[v].push_back(u);
    }
    dfs1(1,0);
    dfs2(1,0,1);
    build(1,1,n);
}

inline void mainwork()
{
    int op,u,v;
    for(int i=0;i<m;i++)
    {
        scanf("%d%d%d",&op,&u,&v);
        if(op==0)
            updatework(u,v);
        else
            printf("%lld\n",calsum(u,v));
    }
}

int main()
{
    prework();
    mainwork();
    return 0;
}

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值