[USACO10FEB]慢下来Slowing down
线段树 树的dfs序
来自 洛谷 P1982 的翻译
by GeneralLiu
来自 jzyz 的翻译 %mzx
线段树 dfs序
数据结构的应用
“数据结构 是先有需求 再有应用” by mzx
那么按照这个思路
先看看针对这道题 有什么需求
再考虑用什么数据结构去解决
以及怎么用该数据结构
这是一个树上的题
某个人进了寝室
只会影响到他子树的答案
因为只有他的 子树 回寝室时
要经过他 得slowing down对吧
这时 要对他的 子树的答案全部 区间+1
这是 对dfs序的需求
需要 dfs序 将树转换成区间
区间修改 单点查询 又是对 线段树 的需求
需要 线段树 的高效维护
如有dalao有更高效的方法请博客留言
我目前只学了线段树这个家伙啦
具体应用
dfs序
void dfs(int u){
dfn[u]=++cnt;//dfn[]为树转换为dfs序中的下标
size[u]=1;//u为根的子树大小
int v;
for(int i=head[u];i;i=next[i]){
v=to[i];
if(dfn[v])continue;
dfs(v);
size[u]+=size[v];
}
}
这样一棵子树 就对应了 dfn[]数组 的一段区间
以点k为根的 区间
左端点 是 dfn[k],
右端点 是 dfn[k] + size [k] - 1 。
线段树
main() 函数中的代码 for(int k,i=1;i<=n;i++){
k=read(); //单点查询
printf("%d\n",query(dfn[k],root)); //区间修改
update(dfn[k],dfn[k]+size[k]-1,root);
} 其他函数 void pushdown(int rt){//懒标记下传
if(!add[rt])return;
add[rt<<1]+=add[rt];
add[rt<<1|1]+=add[rt];
add[rt]=0;
}
void update(int x,int y,int l,int r,int rt){
if(x<=l&&r<=y){
add[rt]++;//区间修改时 针对本题 懒标记+1
return;
}
pushdown(rt);
int mid=(l+r)>>1;
if(x<=mid)update(x,y,lson);
if(mid<y)update(x,y,rson);
}
int query(int k,int l,int r,int rt){
//单点查询 所以线段树只用 懒标记add[]数组 即可
if(l==r)return add[rt];
pushdown(rt);
int mid=(l+r)>>1;
if(k<=mid)return query(k,lson);
return query(k,rson);
}
这样就 滋瓷 了本题的修改与查询操作
完
总代码
#include<bits/stdc++.h>
using namespace std;
#define N 100015
#define root 1,n,1
#define lson l,mid,rt<<1
#define rson mid+1,r,rt<<1|1
int n,cnt;
int head[N],next[N<<1],to[N<<1];
int dfn[N],size[N];
int add[N<<2];
int read(){
int ans=0;
char ch=getchar();
for(;!isdigit(ch);ch=getchar());
for(;isdigit(ch);ch=getchar())
ans=(ans<<3)+(ans<<1)+ch-'0';
return ans;
}
void ad(int from,int too){
next[++cnt]=head[from];
to[cnt]=too;
head[from]=cnt;
}
void dfs(int u){
dfn[u]=++cnt;//dfn[]为树转换为dfs序中的下标
size[u]=1;//u为根的子树大小
int v;
for(int i=head[u];i;i=next[i]){
v=to[i];
if(dfn[v])continue;
dfs(v);
size[u]+=size[v];
}
}
void pushdown(int rt){//懒标记下传
if(!add[rt])return;
add[rt<<1]+=add[rt];
add[rt<<1|1]+=add[rt];
add[rt]=0;
}
void update(int x,int y,int l,int r,int rt){
if(x<=l&&r<=y){
add[rt]++;//区间修改时 针对本题 懒标记+1
return;
}
pushdown(rt);
int mid=(l+r)>>1;
if(x<=mid)update(x,y,lson);
if(mid<y)update(x,y,rson);
}
int query(int k,int l,int r,int rt){
//单点查询 所以线段树只用 懒标记add[]数组 即可
if(l==r)return add[rt];
pushdown(rt);
int mid=(l+r)>>1;
if(k<=mid)return query(k,lson);
return query(k,rson);
}
int main(){
n=read();
for(int x,y,i=1;i<n;i++){
x=read(),y=read();
ad(x,y);
ad(y,x);
}
cnt=0;
dfs(1);
for(int k,i=1;i<=n;i++){
k=read(); //单点查询
printf("%d\n",query(dfn[k],root)); //区间修改
update(dfn[k],dfn[k]+size[k]-1,root);
}
return 0;
}
[USACO10FEB]慢下来Slowing down的更多相关文章
- USACO10FEB]慢下来Slowing down dfs序 线段树
[USACO10FEB]慢下来Slowing down 题面 洛谷P2982 本来想写树剖来着 暴力数据结构直接模拟,每头牛回到自己的农场后,其子树下的所有牛回到农舍时,必定会经过此牛舍,即:每头牛回 ...
- 洛谷P2982 [USACO10FEB]慢下来Slowing down [2017年四月计划 树状数组01]
P2982 [USACO10FEB]慢下来Slowing down 题目描述 Every day each of Farmer John's N (1 <= N <= 100,000) c ...
- 洛谷P2982 [USACO10FEB]慢下来Slowing down(线段树 DFS序 区间增减 单点查询)
To 洛谷.2982 慢下来Slowing down 题目描述 Every day each of Farmer John's N (1 <= N <= 100,000) cows con ...
- [luogu2982][USACO10FEB]慢下来Slowing down(树状数组+dfs序)
题目描述 Every day each of Farmer John's N (1 <= N <= 100,000) cows conveniently numbered 1..N mov ...
- 洛谷P2982 [USACO10FEB]慢下来Slowing down
题目 题目大意 :给出一棵树,节点有点权,求每个节点的祖先中点权小于该节点的结点的个数 . 思路如下 : 从根节点开始,对树进行深度优先遍历. 当进行到节点 i 时,有: $\text{i}$ 的祖 ...
- luoguP2982 [USACO10FEB]慢下来Slowing down
https://www.luogu.org/problemnew/show/P2982 这题你写个树剖当然可以做,但是我们还有一种更简单的方法,使用 dfs 序 + 树状数组即可 考虑一只牛到了自己的 ...
- 线段树+Dfs序【p2982】[USACO10FEB]慢下来Slowing down
Description 每天Farmer John的N头奶牛(1 <= N <= 100000,编号1-N)从粮仓走向他的自己的牧场.牧场构成了一棵树,粮仓在1号牧场.恰好有N-1条道路直 ...
- 洛谷 P2982 [USACO10FEB]慢下来Slowing down
题目描述 Every day each of Farmer John's N (1 <= N <= 100,000) cows conveniently numbered 1..N mov ...
- [luoguP2982][USACO10FEB]慢下来Slowing down(dfs序 + 线段树)
传送门 这个题显然可以用树链剖分做. 然而线段树也能做. 每个点都对它的子树有贡献,所以先求一边 dfs序,然后直接在 dfs序 中搞 线段树 就行. ——代码 #include <cstdio ...
随机推荐
- Backbone.js入门教程第二版笔记(1)
1.模块 集合 视图 和事件的一个综合例子 <!DOCTYPE html> <html> <head> <meta charset="UTF-8&q ...
- MongoDB管理练习
一.索引 1.插入10W条数据 文档内容为:{name:zs-i,age:1} 2016-06-07T14:35:57.041+0800 I CONTROL [initandlisten] > ...
- Sublime3注册码和安装中文包
1.Sublime3注册码 在工具栏Help中点击Enter license,粘贴下面一大串 —– BEGIN LICENSE —– Michael Barnes Single User Licens ...
- AJPFX:不用递归巧妙求出1000的阶乘所有零和尾部零的个数
package com.jonkey.test; import java.math.BigInteger; public class Test6 { /*** @param args* 需求:求出1 ...
- 【转】JobScheduler
JobScheduler JobScheduler是Android L(API21)新增的特性,用于定义满足某些条件下执行的任务.它的宗旨是把一些不是特别紧急的任务放到更合适的时机批量处理,这样可以有 ...
- Bundle的用法
一.API文档说明 1.介绍 用于不同Activity之间的数据传递 1.重要方法 clear():清除此Bundle映射中的所有保存的数据. clone():克隆当前Bundle containsK ...
- ubuntu下安装mongo扩展
安装openssl apt-get install openssl libssl-dev libssl0.9.8 libgtk2.0-dev 安装php-pear apt-get install ph ...
- windows系统下查看或删除自己电脑的共享文件以及文件夹
(1)查看所有共享 net share (2)删除指定共享 例如:删除C盘共享 net share C$ /delete net share 共享名 /delete (/del)
- SAP CRM点了附件的超链接后报错的处理方式
SAP CRM系统里,点击了附件的这些超链接后,如果是文本文件,会在浏览器里打开.如果是其他类型的文件,会弹出下载对话框. 然而最近我工作时遇到一个问题,点击超链接后,总是弹出Logon failed ...
- Python3简明教程(三)—— 运算符和表达式
运算符 什么是运算符? 举个简单的例子 4 +5 = 9 . 例子中,4 和 5 被称为操作数,"+" 称为运算符. Python支持以下类型的运算符: 算术运算符 关系运算符 赋 ...