CF916E Jamie and Tree

题意翻译

有一棵n个节点的有根树,标号为1-n,你需要维护以下三种操作

1.给定一个点v,将整颗树的根变为v

2.给定两个点u, v,将lca(u, v)所在的子树都加上x

3.给定一个点v,你需要回答以v所在的子树的权值和

Translated by mangoyang


错误日志: 第一次 \(debug\) 是 \(jump\) 数组第二维开小了; 交了一次错了, 第二次没有特判修改/查询节点等于根的情况; 第三次 \(RE\) 又是数组开销了 。。。空间那么大我倒是把数组卡大点啊啊啊


Solution

树链剖分, 要求换根修改和查询

\(lca(u,v)\) 等于 \(lca(u, v)\ ,lca(u, root)\ ,lca(v, root)\) 里深度最大的点

修改和查询: 分三种情况考虑:

  1. 操作节点为根节点: 直接操作于整棵树
  2. 根节点在操作节点子树之外: 直接操作即可
  3. 根节点位于操作节点子树内: 利用容斥(最好画图看看)。设点 \(son\) 为从根节点到操作节点路径上的倒数第二个点,先整棵树更新, 再将 \(son\) 的子树减去更新值即可

    (又或者常数较大的先更新整棵树, 反过来减去操作节点的子树, 再更新操作节点这一个点)

Code

#include<iostream>
#include<cstdio>
#include<queue>
#include<cstring>
#include<algorithm>
typedef long long LL;
using namespace std;
LL RD(){
LL out = 0,flag = 1;char c = getchar();
while(c < '0' || c >'9'){if(c == '-')flag = -1;c = getchar();}
while(c >= '0' && c <= '9'){out = out * 10 + c - '0';c = getchar();}
return flag * out;
}
const LL maxn = 200019,INF = 1e9 + 19;
LL head[maxn],nume = 1;
struct Node{
LL v,dis,nxt;
}E[maxn << 2];
void add(LL u,LL v,LL dis){
E[++nume].nxt = head[u];
E[nume].v = v;
E[nume].dis = dis;
head[u] = nume;
}
LL num, na;
LL dep[maxn], size[maxn], fa[maxn], wson[maxn], top[maxn], pos[maxn], ori[maxn], cnt;
LL v[maxn];
void dfs1(LL id, LL F){
size[id] = 1;
for(LL i = head[id];i;i = E[i].nxt){
LL v = E[i].v;
if(v == F)continue;
dep[v] = dep[id] + 1;
fa[v] = id;
dfs1(v, id);
size[id] += size[v];
if(size[v] > size[wson[id]])wson[id] = v;
}
}
void dfs2(LL id, LL TP){
pos[id] = ++cnt;
ori[cnt] = id;
top[id] = TP;
if(!wson[id])return ;
dfs2(wson[id], TP);
for(LL i = head[id];i;i = E[i].nxt){
LL v = E[i].v;
if(v == fa[id] || v == wson[id])continue;
dfs2(v, v);
}
}
#define lid (id << 1)
#define rid (id << 1) | 1
struct seg_tree{
LL l, r;
LL lazy, sum;
}tree[maxn << 2];
void pushup(LL id){tree[id].sum = tree[lid].sum + tree[rid].sum;}
void pushdown(LL id){
if(tree[id].lazy){
LL val = tree[id].lazy;
tree[lid].lazy += val;
tree[rid].lazy += val;
tree[lid].sum += (tree[lid].r - tree[lid].l + 1) * val;
tree[rid].sum += (tree[rid].r - tree[rid].l + 1) * val;
tree[id].lazy = 0;
}
}
void build(LL id, LL l, LL r){
tree[id].l = l, tree[id].r = r;
if(l == r){
tree[id].sum = v[ori[l]];
return ;
}
LL mid = (l + r) >> 1;
build(lid, l, mid), build(rid, mid + 1, r);
pushup(id);
}
void update(LL id, LL val, LL l, LL r){
pushdown(id);
if(tree[id].l == l && tree[id].r == r){
tree[id].sum += (tree[id].r - tree[id].l + 1) * val;
tree[id].lazy += val;
return ;
}
LL mid = (tree[id].l + tree[id].r) >> 1;
if(mid < l)update(rid, val, l, r);
else if(mid >= r)update(lid, val, l, r);
else update(lid, val, l, mid), update(rid, val, mid + 1, r);
pushup(id);
}
LL query(LL id, LL l ,LL r){
pushdown(id);
if(tree[id].l == l && tree[id].r == r){
return tree[id].sum;
}
LL mid = (tree[id].l + tree[id].r) >> 1;
if(mid < l)return query(rid, l, r);
else if(mid >= r)return query(lid, l, r);
else return query(lid, l, mid) + query(rid, mid + 1, r);
}
LL root = 1, jump[maxn][25];
void get_jump(){
for(LL i = 1;i <= num;i++)jump[i][0] = fa[i];
for(LL i = 1;i <= 19;i++){
for(LL j = 1;j <= num;j++){
jump[j][i] = jump[jump[j][i - 1]][i - 1];
}
}
}
LL LCA(LL x, LL y){
if(dep[x] < dep[y])swap(x, y);
for(LL i = 19;i >= 0;i--)if(dep[jump[x][i]] >= dep[y])x = jump[x][i];
if(x == y)return x;
for(LL i = 19;i >= 0;i--)if(jump[x][i] != jump[y][i])x = jump[x][i], y = jump[y][i];
return jump[x][0];
}
LL real_LCA(LL x, LL y){
LL lca1 = LCA(x, y), lca2 = LCA(x, root), lca3 = LCA(y, root);
LL lca = dep[lca1] > dep[lca2] ? lca1 : lca2;
return dep[lca] > dep[lca3] ? lca : lca3;
}
LL son_LCA(LL x, LL y = root){
if(dep[x] >= dep[y])return -1;
for(LL i = 19;i >= 0;i--)if(dep[jump[y][i]] >= dep[x] + 1)y = jump[y][i];
if(fa[y] == x)return y;
return -1;
}
void change_root(){root = RD();}
void uprange(){
LL x = RD(), y = RD(), val = RD();
LL lca = real_LCA(x, y);
if(lca == root){update(1, val, pos[1], pos[1] + size[1] - 1);return ;}
LL son = son_LCA(lca);
if(son == -1){update(1, val, pos[lca], pos[lca] + size[lca] - 1);return ;}
update(1, val, pos[1], pos[1] + size[1] - 1);
update(1,-val, pos[son], pos[son] + size[son] - 1);
}
void get_sum(){
LL x = RD();
if(x == root){printf("%lld\n", query(1, pos[1], pos[1] + size[1] - 1));return ;}
LL son = son_LCA(x);
if(son == -1){printf("%lld\n", query(1, pos[x], pos[x] + size[x] - 1));return ;}
printf("%lld\n", query(1, pos[1], pos[1] + size[1] - 1) - query(1, pos[son], pos[son] + size[son] - 1));
}
int main(){
num = RD();na = RD();
for(LL i = 1;i <= num;i++)v[i] = RD();
for(LL i = 1;i <= num - 1;i++){
LL u = RD(), v = RD();
add(u, v, 1), add(v, u, 1);
}
dep[1] = 1;
dfs1(1, -1), dfs2(1, 1);
build(1, 1, num);
get_jump();
for(LL i = 1;i <= na;i++){
LL cmd = RD();
if(cmd == 1)change_root();
else if(cmd == 2)uprange();
else get_sum();
}
return 0;
}

CF916E Jamie and Tree的更多相关文章

  1. CF916E Jamie and Tree 解题报告

    CF916E Jamie and Tree 题意翻译 有一棵\(n\)个节点的有根树,标号为\(1-n\),你需要维护一下三种操作 1.给定一个点\(v\),将整颗树的根变为\(v\) 2.给定两个点 ...

  2. 题解 [CF916E] Jamie and Tree

    题面 解析 这题考试时刚了四个小时. 结果还是爆零了 主要就是因为\(lca\)找伪了. 我们先考虑没有操作1,那就是裸的线段树. 在换了根以后,主要就是\(lca\)不好找(分类讨论伪了). 我们将 ...

  3. 【树剖】CF916E Jamie and Tree

    好吧这其实应该不是树剖... 因为只要求子树就够了,dfs就好了 大概就是记录一个全局根root 多画几幅图会发现修改时x,y以root为根时的lca为以1为根时的lca(x,y),lca(root, ...

  4. Codeforces 916E Jamie and Tree (换根讨论)

    题目链接  Jamie and Tree 题意  给定一棵树,现在有下列操作: $1$.把当前的根换成$v$:$2$.找到最小的同时包含$u$和$v$的子树,然后把这棵子树里面的所有点的值加$x$: ...

  5. codeforces 916E Jamie and Tree dfs序列化+线段树+LCA

    E. Jamie and Tree time limit per test 2.5 seconds memory limit per test 256 megabytes input standard ...

  6. CodeForces 916E Jamie and Tree(树链剖分+LCA)

    To your surprise, Jamie is the final boss! Ehehehe. Jamie has given you a tree with n vertices, numb ...

  7. Jamie and Tree CodeForces - 916E (换根)

    大意: n节点树, 每个点有权值, 三种操作: 1,换根. 2, lca(u,v)的子树权值全部增加x. 3, 查询子树权值和. 先不考虑换根, 考虑子树x加v的贡献 (1)对fa[x]到根的树链贡献 ...

  8. Jamie and Tree (dfs序 + 最近公共祖先LCA)

    题面 题解 我们求它子树的权值和,一般用dfs序把树拍到线段树上做. 当它换根时,我们就直接把root赋值就行了,树的结构不去动它. 对于第二个操作,我们得到的链和根的相对位置有三种情况: 设两点为A ...

  9. CF916E

    Codeforces 916E 简要题解Description Description 有一棵n个点的树,每个节点上有一个权值wi,最开始根为1号点.现在有3种类型的操作: 1 root, 表示将根设 ...

随机推荐

  1. BugPhobia展示篇章:学霸在线系统Alpha阶段展示

    0x00:序言 1 universe, 9 planets, 204 countries,809 islands, 7 seas, and i had the privilege to meet yo ...

  2. 福大软工 Alpha 事后诸葛亮

    写在前面 林燊大哥 一路走来,好不容易,终于完结了. 设想和目标 1. 我们的软件要解决什么问题?是否定义得很清楚?是否对典型用户和典型场景有清晰的描述? 解决的问题 用户在进店之前无法得知店铺的优劣 ...

  3. struts2 jsp提交对象数据要这么干

    不要每个属性都 setter getter .. 这样页面很难看... 直接 把对象变成一个成员变量会比较好. Java code ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 ...

  4. Internet History, Technology and Security (Week8)

    Week 8 This week we start two weeks of Internet Security. It is a little technical but don't worry - ...

  5. Unity如何判断网络状态?

    根据Application.internetReachability来判断网络状态 NetworkReachability.NotReachable 网络不可用 NetworkReachability ...

  6. 今年暑假要AC

    今年暑假要AC 在这个大学的第一个的暑假,谁不想回去high呢~ 但是,这是不行的,还没有AC,你能回去吗?高三那年的暑假怎么玩的,现在补回来吧...有规模有计划有氛围的学习就是:优点多效率好激情足~ ...

  7. visual stdio2013软件安装及单元测试

    visual stdio2013软件安装及单元测试 一.visual stdio2013软件安装详解 今天,笔者为大家带来如何在Windows10下完美安装Visual Studio 2013专业版. ...

  8. 软工网络15团队作业8——Beta阶段敏捷冲刺(day1)

    第 1 篇 Scrum 冲刺博客 1. 介绍小组新加入的成员,Ta担任的角色 --给出让ta担当此角色的理由 小组新加入的成员:3085叶金蕾 担任的角色:测试/用户体验/开发 理由:根据小组讨论以及 ...

  9. 爬虫学习之-scrapy交互式命令

    scrapy shell https:///www.baidu.com  会启动爬虫请求网页 view(response) 会在浏览器打开请求到的临时文件 response.xpath("/ ...

  10. webservice(二)简单实例

    1.建立WSDL文件      建立WSDL的工具很多,eclipse.zendstudio.vs都可以,我个人建议自己写,熟悉结构,另外自动工具对xml schame类型支持在类型中可能会报错. 下 ...