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;
}