hdu 3966 (树链剖分+线段树)

题意:给出一棵树,每个节点有一些敌人,有三种操作,I:x,y,路径上的所有点的人数+w。D:x,y,路径上的所有点的人数-w。Q:节点x的人数。

思路:如果是线性的线段树可以做,所以要树链剖分,剖分后的每条链是线性的,更改区间时用找公共祖先的方式,将两点找到同一条链上,同一条链上的两点间可以直接线段树更改。






#pragma comment(linker, "/STACK:1024000000,1024000000")
#include<stdio.h>
#include<string.h>
#include <iostream>
using namespace std;
const int N=51000;
int son[N],sz[N],dep[N],top[N],father[N],head[N],ti[N],num,idx,a[N];
struct edge
{
	int ed,next;
}e[N*4];
void addedge(int x,int y)
{
	e[num].ed=y;e[num].next=head[x];head[x]=num++;
	e[num].ed=x;e[num].next=head[y];head[y]=num++;
}
//*******************树链剖分**********************
void find_son(int u)
{
	int i,v;
	sz[u]=1;son[u]=0;
	for(i=head[u];i!=-1;i=e[i].next)
	{
		v=e[i].ed;
		if(v==father[u])continue;
		father[v]=u;dep[v]=dep[u]+1;
		find_son(v);
		sz[u]+=sz[v];
		if(sz[v]>sz[son[u]])son[u]=v;
	}
}
void find_time(int u,int fa)
{
	int i,v;
	ti[u]=idx++;	
	top[u]=fa;
	if(son[u]!=0)find_time(son[u],top[u]);
	for(i=head[u];i!=-1;i=e[i].next)
	{
		v=e[i].ed;
		if(v==son[u]||v==father[u])continue;
		find_time(v,v);
	}
}
//************************线段树**************************
struct Tree
{
	int R,L,ct;//ct:该区间的增量
}T[N*4];
void buildTree(int L,int R,int id)
{
	T[id].L=L;T[id].R=R;T[id].ct=0;
	if(L==R) return ;		
	int mid=(L+R)>>1,li=id<<1,ri=li+1;
	buildTree(L,mid,li);
	buildTree(mid+1,R,ri);
}
void insert(int L,int R,int id,int w)
{
	if(T[id].L==L&&T[id].R==R)
	{
		T[id].ct+=w;
		return ;
	}
	int mid=(T[id].L+T[id].R)>>1,li=id<<1,ri=li+1;
	if(mid>=R)insert(L,R,li,w);
	else if(mid<L)insert(L,R,ri,w);
	else 
	{
		insert(L,mid,li,w);
		insert(mid+1,R,ri,w);
	}
}
int find(int i,int id,int w)
{
	if(T[id].L==T[id].R)
	    return w+T[id].ct;
	int mid=(T[id].L+T[id].R)>>1,li=id<<1,ri=li+1;
	if(mid>=i)return find(i,li,w+T[id].ct);
	else return find(i,ri,w+T[id].ct);
}
void lca(int x,int y,int w)
{
	while(top[x]!=top[y])
	{
		if(dep[top[x]]<dep[top[y]])swap(x,y);
		insert(ti[top[x]],ti[x],1,w);
		x=father[top[x]];
	}
	if(dep[x]>dep[y])swap(x,y);
	insert(ti[x],ti[y],1,w);
}
int main()
{
	int i,n,m,q,x,y,w;
	char str[10];
	while(scanf("%d%d%d",&n,&m,&q)!=-1)
	{
		memset(head,-1,sizeof(head));
		num=0;
		for(i=1;i<=n;i++)
			scanf("%d",&a[i]);
		for(i=1;i<=m;i++)
		{
			scanf("%d%d",&x,&y);
			addedge(x,y);
		}
		father[1]=sz[0]=0;dep[1]=idx=1;
		find_son(1);
		find_time(1,1);
		buildTree(1,n,1);
		while(q--)
		{
			scanf("%s",str);
			if(str[0]=='Q')
			{
				scanf("%d",&x);
				printf("%d\n",a[x]+find(ti[x],1,0));
			}
			else
			{
				scanf("%d%d%d",&x,&y,&w);
				if(str[0]=='I')	w=-w;				
				lca(x,y,-w);
			}
		}
	}
	return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值