题目描述

Farmer John has N barren pastures (2 <= N <= 100,000) connected by N-1 bidirectional roads, such that there is exactly one path between any two pastures. Bessie, a cow who loves her grazing time, often complains about how there is no grass on the roads between pastures. Farmer John loves Bessie very much, and today he is finally going to plant grass on the roads. He will do so using a procedure consisting of M steps (1 <= M <= 100,000).

At each step one of two things will happen:

  • FJ will choose two pastures, and plant a patch of grass along each road in between the two pastures, or,

  • Bessie will ask about how many patches of grass on a particular road, and Farmer John must answer her question.

Farmer John is a very poor counter -- help him answer Bessie's questions!

给出一棵n个节点的树,有m个操作,操作为将一条路径上的边权加一或询问某条边的权值。

输入输出格式

输入格式:

  • Line 1: Two space-separated integers N and M

  • Lines 2..N: Two space-separated integers describing the endpoints of a road.

  • Lines N+1..N+M: Line i+1 describes step i. The first character of the line is either P or Q, which describes whether or not FJ is planting grass or simply querying. This is followed by two space-separated integers A_i and B_i (1 <= A_i, B_i <= N) which describe FJ's action or query.

输出格式:

  • Lines 1..???: Each line has the answer to a query, appearing in the same order as the queries appear in the input.

输入输出样例

输入样例#1:

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
输出样例#1:

2
1
2

树剖裸题

链上修改+查询

屠龙宝刀点击就送

#include <ctype.h>
#include <cstdio>
#define M 100005
void read(int &x)
{
x=;register char ch=getchar();
for(;!isdigit(ch);ch=getchar());
for(;isdigit(ch);ch=getchar()) x=x*+ch-'';
}
struct Edge
{
int next,to;
Edge (int next=,int to=):next(next),to(to) {}
}edge[M<<];
int n,m,head[M],cnt,top[M],belong[M],tim,size[M],dad[M],dep[M];
void insert(int u,int v)
{
edge[++cnt]=Edge(head[u],v);
head[u]=cnt;
}
void swap(int &x,int &y)
{
int tmp=y;
y=x;
x=tmp;
}
struct node
{
int mid,l,r,dis,flag;
node *left,*right;
node ()
{
left=right=NULL;
dis=flag=;
}
}*root;
class t
{
public:
void pushup(node *&k)
{
k->dis=k->left->dis+k->right->dis;
}
void pushdown(node *&k)
{
if(k->l==k->r) return;
k->left->flag+=k->flag;
k->right->flag+=k->flag;
k->left->dis+=k->flag*(k->left->r-k->left->l+);
k->right->dis+=k->flag*(k->right->r-k->right->l+);
k->flag=;
}
void build(node *&k,int l,int r)
{
k=new node;
k->l=l;k->r=r;k->mid=(l+r)>>;
if(l==r) return;
build(k->left,l,k->mid);
build(k->right,k->mid+,r);
}
void Tree_change(node *&k,int l,int r)
{
if(k->l==l&&k->r==r)
{
k->flag++;
k->dis+=(r-l+);
return;
}
if(l>k->mid) Tree_change(k->right,l,r);
else if(r<=k->mid) Tree_change(k->left,l,r);
else Tree_change(k->left,l,k->mid),Tree_change(k->right,k->mid+,r);
pushup(k);
}
int Tree_query(node *&k,int l,int r)
{
if(k->l==l&&k->r==r) return k->dis;
if(k->flag) pushdown(k);
if(l>k->mid) return Tree_query(k->right,l,r);
else if(r<=k->mid) return Tree_query(k->left,l,r);
else return Tree_query(k->left,l,k->mid)+Tree_query(k->right,k->mid+,r);
pushup(k);
}
};
class t Tree;
class sp
{
public :
void dfs1(int x)
{
size[x]=;
dep[x]=dep[dad[x]]+;
for(int i=head[x];i;i=edge[i].next)
{
int v=edge[i].to;
if(dad[x]!=v)
{
dad[v]=x;
dfs1(v);
size[x]+=size[v];
}
}
}
void dfs2(int x)
{
int pos=;
belong[x]=++tim;
if(!top[x]) top[x]=x;
for(int i=head[x];i;i=edge[i].next)
{
int v=edge[i].to;
if(dad[x]!=v&&size[pos]<size[v]) pos=v;
}
if(pos) top[pos]=top[x],dfs2(pos);
for(int i=head[x];i;i=edge[i].next)
{
int v=edge[i].to;
if(dad[x]!=v&&v!=pos) dfs2(v);
}
}
void Chain_change(int x,int y)
{
for(;top[x]!=top[y];x=dad[top[x]])
{
if(dep[top[x]]<dep[top[y]]) swap(x,y);
Tree.Tree_change(root,belong[top[x]],belong[x]);
}
if(x==y) return;
if(dep[x]>dep[y]) swap(x,y);
Tree.Tree_change(root,belong[x]+,belong[y]);
}
int Chain_query(int x,int y)
{
int ans=;
for(;top[x]!=top[y];x=dad[top[x]])
{
if(dep[top[x]]<dep[top[y]]) swap(x,y);
ans+=Tree.Tree_query(root,belong[top[x]],belong[x]);
}
if(x==y) return ans;
if(dep[x]>dep[y]) swap(x,y);
ans+=Tree.Tree_query(root,belong[x]+,belong[y]);
return ans;
}
};
class sp Chain;
int main()
{
read(n);
read(m);
for(int x,y,i=;i<n;i++)
{
read(x);
read(y);
insert(x,y);
insert(y,x);
}
Chain.dfs1();Chain.dfs2();
char str[];
root=new node;
Tree.build(root,,n);
for(int x,y;m--;)
{
scanf("%s",str+);
read(x);
read(y);
if(str[]=='P') Chain.Chain_change(x,y);
else printf("%d\n",Chain.Chain_query(x,y));
}
return ;
}

洛谷 P3038 [USACO11DEC]牧草种植Grass Planting的更多相关文章

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

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

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

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

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

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

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

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

  5. 树链剖分【p3038】[USACO11DEC]牧草种植Grass Planting

    表示看不太清. 概括题意 树上维护区间修改与区间和查询. 很明显树剖裸题,切掉,细节处错误T了好久 TAT 代码 #include<cstdio> #include<cstdlib& ...

  6. [USACO11DEC]牧草种植Grass Planting

    图很丑.明显的树链剖分,需要的操作只有区间修改和区间查询.不过这里是边权,我们怎么把它转成点权呢?对于E(u,v),我们选其深度大的节点,把边权扔给它.因为这是树,所以每个点只有一个父亲,所以每个边权 ...

  7. 【LuoguP3038/[USACO11DEC]牧草种植Grass Planting】树链剖分+树状数组【树状数组的区间修改与区间查询】

    模拟题,可以用树链剖分+线段树维护. 但是学了一个厉害的..树状数组的区间修改与区间查询.. 分割线里面的是转载的: ----------------------------------------- ...

  8. 洛谷P3038 牧草种植Grass Planting

    思路: 首先,这道题的翻译是有问题的(起码现在是),查询的时候应该是查询某一条路径的权值,而不是某条边(坑死我了). 与平常树链剖分题目不同的是,这道题目维护的是边权,而不是点权,那怎么办呢?好像有点 ...

  9. 洛谷P3038 牧草种植 [树链剖分]

    题目传送门 牧草种植 题目描述 Farmer John has N barren pastures (2 <= N <= 100,000) connected by N-1 bidirec ...

随机推荐

  1. SPOJ:Elegant Permuted Sum(贪心)

    Special Thanks: Jane Alam Jan*At moment in University of Texas at San Antonio - USA You will be give ...

  2. Excel: 应用Match/Vlookup比较Excel两列的不同数据

    假设Excel中有两列,现在要比较两列数据的不同.

  3. struct 结构体解析(原)

    (一)基本概念 结构体是一个或是多个变量的集合,这些变量可能为不同的类型,为了处理的方便而将这些变量组合在一个名字之下.我们将关键字struct引入了结构声明中.结构声明包含在花括号内的一系列声明组成 ...

  4. python的time模块使用

    在平常的代码中,我们常常需要与时间打交道.在Python中,与时间处理有关的模块就包括:time,datetime以及calendar.这篇文章,主要讲解time模块. 在开始之前,首先要说明这几点: ...

  5. std::unique

    类属性算法unique的作用是从输入序列中“删除”所有相邻的重复元素. 该算法删除相邻的重复元素,然后重新排列输入范围内的元素,并且返回一个迭代器(容器的长度没变,只是元素顺序改变了),表示无重复的值 ...

  6. Mysql数据库实现高可用

    Mysql实现高可用 MMM MMM(master-master replication manager for mysql)mysql主主复制管理器. MMM是一套灵活的脚本程序,基于perl实现, ...

  7. 任务26:dotnet watch run 和attach到进程调试

    任务26:dotnet watch run 和attach到进程调试 dotnet watch run 的一种调试方法 打开VSCode,先关闭当前的文件夹 Ctrl+~快捷键 打开窗体. ls应该是 ...

  8. 简析hotjar录屏功能实现原理

    简析hotjar录屏功能实现原理 众所周知,hotjar中录屏功能是其重要的一个卖点,看着很牛X酷炫的样子,今天就简单的分析一下其可能实现(这里只根据其请求加上个人理解分析,并不代表hotjar中真实 ...

  9. 土法搞docker系列之自制docker的graph driver vdisk

    写在最前 偶然整理,翻出来14年刚开始学docker的时候的好多资料.当时docker刚刚进入国内,还有很多的问题.当时我们的思考方式很简单,docker确实是个好的工具,虽然还不成熟.但是不能因为短 ...

  10. 报错org.springframework.dao.DataIntegrityViolationException

    最简单的原因可能是数据库外键字段选择了不能为空, 改为允许为空就行了.