大致题意: 维护一棵树,支持两种操作:

P x y x到y路径上的每条边的值+1;
Q x y 询问x到y路径上所有边的值的和。
Input
第一行两个正整数,N,M表示点数和操作数;
接下来N-1行每行两个数表示一条边;
接下来M行表示M个操作,每行形如P x y或Q x y。
2≤N≤100,000,1≤M≤100,000。
Output
M行,对应相应询问的答案。
Sample Input
4 6
1 4
2 4
3 4
P 2 3
P 1 3
Q 3 4
P 1 4
Q 2 4
Q 1 4
Sample Output
2
1
2

/*
大致题意: 维护一棵树,支持两种操作:
P x y x到y路径上的每条边的值+1;
Q x y 询问x到y路径上所有边的值的和
*/
#include<cstdio>
#include<algorithm>
using namespace std;
const int maxn=100050;
int m;
int read()
{
int x=0,f=1;char ch=getchar();
for(;ch<'0'||ch>'9';ch=getchar())if(ch=='-')f=-1;
for(;ch>='0'&&ch<='9';ch=getchar())x=(x<<1)+(x<<3)+ch-'0';
return x*f;
}
struct segment_tree
{
int sum[maxn*8],lazy[maxn*8];
void updata(int p){sum[p]=sum[p<<1]+sum[p<<1|1];}
void add_tag(int p,int v)
{
lazy[p]+=v;
sum[p]+=v;
}
void pushdown(int p)
{
if(!lazy[p])return;
add_tag(p<<1,lazy[p]);
add_tag(p<<1|1,lazy[p]);
lazy[p]=0;
}
void add(int p,int l,int r,int x,int y)
{
if(x>y)return;
if(x<=l&&r<=y)
{
add_tag(p,1);
return;
}
int mid=(l+r)>>1;
pushdown(p);
if(x<=mid)
add(p<<1,l,mid,x,y);
if(y>mid)
add(p<<1|1,mid+1,r,x,y);
updata(p);
}
int query(int p,int l,int r,int x,int y)
{
if(x>y)return 0;
if(x<=l&&r<=y)
return sum[p];
int mid=(l+r)>>1,ls=0,rs=0;
pushdown(p);
if(x<=mid)
ls=query(p<<1,l,mid,x,y);
if(y>mid)
rs=query(p<<1|1,mid+1,r,x,y);
return ls+rs;
}
}ST;
struct Tree
{
int n,tot,cnt;
int pre[maxn*2],son[maxn*2],now[maxn];
int dep[maxn],fa[maxn],siz[maxn],wson[maxn],top[maxn],seg[maxn];
void add(int a,int b)
{
pre[++tot]=now[a];
now[a]=tot;
son[tot]=b;
}
void in()
{
n=read(),m=read();
for(int i=1;i<n;i++)
{
int x=read(),y=read();
add(x,y);
add(y,x);
}
}
void dfs(int faa,int u)
{
fa[u]=faa;
dep[u]=dep[faa]+1;
siz[u]=1;
for(int p=now[u],v=son[p];p;p=pre[p],v=son[p])
if(v!=faa)
{
dfs(u,v);
siz[u]+=siz[v];
if(siz[wson[u]]<siz[v])wson[u]=v;
}
}
void base(int tp,int u) //树链剖分
{
top[u]=tp;
seg[u]=++cnt;
if(wson[u])
base(tp,wson[u]);
for(int p=now[u],v=son[p];p;p=pre[p],v=son[p])
if(v!=fa[u]&&v!=wson[u])base(v,v);
}
void addedg(int a,int b) //a到b的路径上边都加1
{
while(top[a]!=top[b])
{
if(dep[top[a]]<dep[top[b]])
swap(a,b);
ST.add(1,1,n,seg[top[a]],seg[a]);
a=fa[top[a]];
}
if(dep[a]<dep[b])
swap(a,b);
ST.add(1,1,n,seg[b]+1,seg[a]);
//注意是对边操作,所以是seg[b]+1
}
void query(int a,int b)
{
int ans=0;
while(top[a]!=top[b])
{
if(dep[top[a]]<dep[top[b]])swap(a,b);
ans+=ST.query(1,1,n,seg[top[a]],seg[a]);
a=fa[top[a]];
}
if(dep[a]<dep[b])
swap(a,b);
ans+=ST.query(1,1,n,seg[b]+1,seg[a]);
printf("%d\n",ans);
}
}CT;
void work()
{
for(int i=1;i<=m;i++)
{
char s[10];int x,y;
scanf("%s",s);
x=read(),y=read();
if(s[0]=='P') //路径上加1
CT.addedg(x,y);
else
CT.query(x,y);
}
}
int main()
{
CT.in();
CT.dfs(0,1);
CT.base(1,1);
work();
return 0;
}

  

Grass Planting的更多相关文章

  1. spoj - Grass Planting(树链剖分模板题)

    Grass Planting 题意 给出一棵树,树有边权.每次给出节点 (u, v) ,有两种操作:1. 把 u 到 v 路径上所有边的权值加 1.2. 查询 u 到 v 的权值之和. 分析 如果这些 ...

  2. USACO Grass Planting

    洛谷 P3038 [USACO11DEC]牧草种植Grass Planting 洛谷传送门 JDOJ 2282: USACO 2011 Dec Gold 3.Grass Planting JDOJ传送 ...

  3. 洛谷P3038 [USACO11DEC]牧草种植Grass Planting

    题目描述 Farmer John has N barren pastures (2 <= N <= 100,000) connected by N-1 bidirectional road ...

  4. [USACO11DEC] Grass Planting (树链剖分)

    题目描述 Farmer John has N barren pastures (2 <= N <= 100,000) connected by N-1 bidirectional road ...

  5. AC日记——[USACO11DEC]牧草种植Grass Planting 洛谷 P3038

    题目描述 Farmer John has N barren pastures (2 <= N <= 100,000) connected by N-1 bidirectional road ...

  6. [Usaco2011 Dec]Grass Planting

    Description Farmer John has N barren pastures connected by N-1 bidirectional roads, such that there ...

  7. 洛谷 P3038 [USACO11DEC]牧草种植Grass Planting

    题目描述 Farmer John has N barren pastures (2 <= N <= 100,000) connected by N-1 bidirectional road ...

  8. P3038 [USACO11DEC]牧草种植Grass Planting

    题目描述 Farmer John has N barren pastures (2 <= N <= 100,000) connected by N-1 bidirectional road ...

  9. 【 SPOJ - GRASSPLA】 Grass Planting (树链剖分+树状数组)

    54  种草约翰有 N 个牧场,编号为 1 到 N.它们之间有 N − 1 条道路,每条道路连接两个牧场.通过这些道路,所有牧场都是连通的.刚开始的时候,所有道路都是光秃秃的,没有青草.约翰会在一些道 ...

  10. 洛谷 P3038 [USACO11DEC]牧草种植Grass Planting(树链剖分)

    题解:仍然是无脑树剖,要注意一下边权,然而这种没有初始边权的题目其实和点权也没什么区别了 代码如下: #include<cstdio> #include<vector> #in ...

随机推荐

  1. Maven搭建简单的SPring+SpringMVC+Hibernate框架

    公司的项目用到的框架是Spring+SpringMVC+Hibernate 以前没有用过,所以要系统的学习一下,首先要学会怎么搭建 第一步  创建一个Maven的web项目  创建方法以前的博客中有提 ...

  2. 使用Spring MVC统一异常处理实战(转载)

    原文地址:http://blog.csdn.net/ufo2910628/article/details/40399539 种方式: (1)使用Spring MVC提供的简单异常处理器SimpleMa ...

  3. java 对象与类

    类与类之间的关系 一.继承关系      继承指的是一个类(称为子类.子接口)继承另外的一个类(称为父类.父接口)的功能,并可以增加它自己的新功能的能力.在Java中继承关系通过关键字extends明 ...

  4. Js中JSON.stringify()与JSON.parse()

    JSON(JavaScript Object Notation)是一种轻量级的数据交换格式.因为采用独立于语言的文本格式,也使用了类似于C语言家族的习惯,拥有了这些特性使使JSON称为理想的数据交换语 ...

  5. Python之面向对象之反射、内置方法

    一.静态方法(staticmethod)和类方法(classmethod) 类方法:有个默认参数cls,并且可以直接用类名去调用,可以与类属性交互(也就是可以使用类属性) 静态方法:让类里的方法直接被 ...

  6. java常用关键词关键字,方法源码解析

    transient volatile native final Integer String Class &&Object newInstance Class.forName,Clas ...

  7. 【UOJ#77】A+B Problem

    传送门 题目描述 略 Sol 看到选择黑白收益不同,然后还可能有代价. 我们想到用网络流解决,并且这应该是用总可能收益-最小割得到答案. 考虑初步建图,发现那个限制可以直接 \(n^2\) 解决. 我 ...

  8. 浅谈Android的资源编译过程

    转载自 http://www.cnblogs.com/dyllove98/p/3144950.html 好长,记录下,一次看完感觉像没看一样 Android APK 一.APK的结构以及生成 APK是 ...

  9. Missing radix parameter.报错解决方法

    当报“Missing radix parameter.”这个错的时候,是因为使用parseInt没有传第二个参数,以前简写的时候大家几乎都不传,甚至不知道还有第二个参数. 当时候Eslint预发检查时 ...

  10. UNIX标准C - 进程之间的通信

    一.基本概念 进程间通信IPC:进程之间交换数据的过程叫进程间通信 进程间同性的方式: 简单的进程间的通信: 命令行:父进程通过exec函数创建子进程是可以附加一些数据 环境变量表:父进程通过exec ...