Grass Planting
大致题意: 维护一棵树,支持两种操作:
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的更多相关文章
- spoj - Grass Planting(树链剖分模板题)
Grass Planting 题意 给出一棵树,树有边权.每次给出节点 (u, v) ,有两种操作:1. 把 u 到 v 路径上所有边的权值加 1.2. 查询 u 到 v 的权值之和. 分析 如果这些 ...
- USACO Grass Planting
洛谷 P3038 [USACO11DEC]牧草种植Grass Planting 洛谷传送门 JDOJ 2282: USACO 2011 Dec Gold 3.Grass Planting JDOJ传送 ...
- 洛谷P3038 [USACO11DEC]牧草种植Grass Planting
题目描述 Farmer John has N barren pastures (2 <= N <= 100,000) connected by N-1 bidirectional road ...
- [USACO11DEC] Grass Planting (树链剖分)
题目描述 Farmer John has N barren pastures (2 <= N <= 100,000) connected by N-1 bidirectional road ...
- AC日记——[USACO11DEC]牧草种植Grass Planting 洛谷 P3038
题目描述 Farmer John has N barren pastures (2 <= N <= 100,000) connected by N-1 bidirectional road ...
- [Usaco2011 Dec]Grass Planting
Description Farmer John has N barren pastures connected by N-1 bidirectional roads, such that there ...
- 洛谷 P3038 [USACO11DEC]牧草种植Grass Planting
题目描述 Farmer John has N barren pastures (2 <= N <= 100,000) connected by N-1 bidirectional road ...
- P3038 [USACO11DEC]牧草种植Grass Planting
题目描述 Farmer John has N barren pastures (2 <= N <= 100,000) connected by N-1 bidirectional road ...
- 【 SPOJ - GRASSPLA】 Grass Planting (树链剖分+树状数组)
54 种草约翰有 N 个牧场,编号为 1 到 N.它们之间有 N − 1 条道路,每条道路连接两个牧场.通过这些道路,所有牧场都是连通的.刚开始的时候,所有道路都是光秃秃的,没有青草.约翰会在一些道 ...
- 洛谷 P3038 [USACO11DEC]牧草种植Grass Planting(树链剖分)
题解:仍然是无脑树剖,要注意一下边权,然而这种没有初始边权的题目其实和点权也没什么区别了 代码如下: #include<cstdio> #include<vector> #in ...
随机推荐
- 第03课:GDB常用的调试命令概览
先给出一个常用命令的列表,后面结合具体的例子详细介绍每个命令的用法. 命令名称 命令缩写 命令说明 run r 运行一个程序 continue c 让暂停的程序继续运行 next n 运行到下 ...
- asp.net mvc + vue.js + axios.js
1.新建一个 MVC 应用程序 2.右键解决方案 添加VUE 3.搜索vue 1.安装axios.js ,用于数据请求,get , post axios
- CodeForces-721B-Passwords
链接: https://vjudge.net/problem/CodeForces-721B 题意: Vanya is managed to enter his favourite site Code ...
- Quartz--Spring 定时任务
一.quartz核心概念 先来看一张图: scheduler 任务调度器 trigger 触发器,用于定义任务调度时间规则 job 任务,即被调度的任务 misfire 错过的,指本来应该被 ...
- [洛谷P3322] SDOI2015 排序
问题描述 小A有一个1-2^N的排列A[1..2^N],他希望将A数组从小到大排序,小A可以执行的操作有N种,每种操作最多可以执行一次,对于所有的 i(1<=i<=N),第i中操作为将序列 ...
- break continue exit return 的区别
[root@localhost day1]# cat ss.sh #!/bin/bash for ((i=0;i<5;i++)) do if [ $i -eq 3 ] then break #c ...
- springmvc请求参数异常统一处理,结合钉钉报告信息定位bug位置
参考之前一篇博客:springmvc请求参数异常统一处理 1.ExceptionHandlerController package com.oy.controller; import java.tex ...
- luoguP1197 [JSOI2008]星球大战 x
P1197 [JSOI2008]星球大战 题目描述 很久以前,在一个遥远的星系,一个黑暗的帝国靠着它的超级武器统治者整个星系.某一天,凭着一个偶然的机遇,一支反抗军摧毁了帝国的超级武器,并攻下了星系中 ...
- Spring Boot教程(二十四)Web应用的统一异常处理
我们在做Web应用的时候,请求处理过程中发生错误是非常常见的情况.Spring Boot提供了一个默认的映射:/error,当处理中抛出异常之后,会转到该请求中处理,并且该请求有一个全局的错误页面用来 ...
- 装RAC跑脚本报错
在执行第二个脚本的时候报错 原因是在改服务器找不到该包 解决方法: 挂载iso镜像,安装这个包