ACM-ICPC 2018 沈阳赛区网络预赛 J. Ka Chang(树上分块+dfs序+线段树)
题意
链接:https://nanti.jisuanke.com/t/A1998
给出一个有根树(根是1),有n个结点。初始的时候每个结点的值都是0.下面有q个操作,操作有两种,操作1.将深度为L(根节点深度为0)的点的值全部增加X。操作2.查询以x为根的子树的结点值得和。其中N,Q<=1e5。
思路
因为这题是对某一深度的所有点加x,所以不是树链剖分。
我们可以先预处理一下dfs序,顺带把d[u]:u的深度、dd[x]:深度为x的点集求出来。
考虑分块,对某一深度分两种情况:1.这一深度的点的个数<=block;2.这一深度的点的个数>block。
对于情况1:更新操作我们可以暴力,因为block我是取的sqrt(n),那么更新的复杂度最坏也是O(qlog(n)sqrt(n)),还能接受,毕竟数据不是特别强。
对于情况2:我们直接将这一层加上x,即add[deep]+=x。
对于查询操作,我们先线段树计算一下这个点为根的子树的值,当然,线段树只暴力更新了size<=block的层,所以查询也只会查询这些层上的值。我们还要计算一下size>block的层,怎么计算呢?
我们先定义一个map<ll,ll> M[N],M[u]表示u为根节点的子树的值的和(size>block的层),M可以通过枚举每个点,如果这个点所在层的size>block,这个点就对它的所有父亲都有贡献,所以往上爬,对父亲加上这个点的贡献。所以size>block的层的值我们可以轻松通过M来得到。
代码
#include<bits/stdc++.h>
using namespace std;
#define inf 0x3f3f3f3f
#define ll long long
#define int ll
const int N=200005;
const double eps=1e-8;
const double PI = acos(-1.0);
struct node
{
    int to,next;
} eg[N];
int head[N],tot=0,sum[N<<2],add[N];
void init()
{
    memset(head,-1,sizeof(head));
    tot=0;
}
void addedge(int u,int v)
{
    eg[tot].to=v;
    eg[tot].next=head[u];
    head[u]=tot++;
}
int in[N],out[N],id=0,q[N],d[N],block,f[N];
vector<int> dd[N];
map<ll,ll> M[N];
void dfs(int x,int fa,int deep)
{
    q[++id]=x;
    in[x]=id;
    d[x]=deep;
    f[x]=fa;
    dd[deep].push_back(x);
    for(int i=head[x]; ~i; i=eg[i].next)
    {
        if(eg[i].to!=fa)
        {
            dfs(eg[i].to,x,deep+1);
        }
    }
    out[x]=id;
}
void pushUp(int rt)
{
    sum[rt]=sum[rt<<1]+sum[rt<<1|1];
}
void build(int l,int r,int rt)
{
    if(l==r)
    {
        sum[rt]=0;
        return;
    }
    int m=(l+r)>>1;
    build(l,m,rt<<1);
    build(m+1,r,rt<<1|1);
    pushUp(rt);
}
void update(int L,int c,int l,int r,int rt)
{
    if(l==r)
    {
        sum[rt]+=c;
        return ;
    }
    int m=(l+r)>>1;
    if(m>=L)    update(L,c,l,m,rt<<1);
    else        update(L,c,m+1,r,rt<<1|1);
    pushUp(rt);
}
int query(int L,int R,int l,int r,int rt)
{
    if(l>=L&&r<=R)
    {
        return sum[rt];
    }
    int m=(l+r)>>1;
    int ans=0;
    if(m>=L)
        ans+=query(L,R,l,m,rt<<1);
    if(m<R)
        ans+=query(L,R,m+1,r,rt<<1|1);
    return ans;
}
ll ask(int rt)
{
    map<ll,ll>::iterator it=M[rt].begin();
    ll res=0;
    for(;it!=M[rt].end();it++)
    {
        res+=(it->second)*add[it->first];
    }
    return res;
}
signed main()
{
    std::ios::sync_with_stdio(false);
    std::cin.tie(0);
    std::cout.tie(0);
    int n,q;
    while(cin>>n>>q)
    {
        id=0;
        memset(in,0,sizeof(in));
        memset(out,0,sizeof(out));
        init();
        for(int i=0; i<n-1; i++)
        {
            int a,b;
            cin>>a>>b;
            addedge(a,b);
            addedge(b,a);
        }
        block=sqrt(n);
        dfs(1,0,0);
        build(1,n,1);
        for(int i=1; i<=n; i++)
        {
            if(dd[d[i]].size()>block)
            {
                int u=i;
                while(u)
                {
                    if(!M[u][d[i]])
                        M[u][d[i]]=1;
                    else
                        M[u][d[i]]++;
                    u=f[u];
                }
            }
        }
    //   for(int i=1;i<=n;i++)
     //       cout<<i<<" "<<in[i]<<" "<<out[i]<<endl;
        while(q--)
        {
            int o,a,b;
            cin>>o>>a;
            if(o==1)
            {
                cin>>b;
                if(dd[a].size()>block)
                {
                    add[a]+=b;
                }
                else
                {
                    int sz=dd[a].size();
                    for(int i=0;i<sz;i++)
                    {
                        int u=dd[a][i];
                        update(in[u],b,1,n,1);
                    }
                }
            }
            else
            {
                cout<<query(in[a],out[a],1,n,1)+ask(a)<<endl;
            }
        }
    }
    return 0;
}
												
											ACM-ICPC 2018 沈阳赛区网络预赛 J. Ka Chang(树上分块+dfs序+线段树)的更多相关文章
- ACM-ICPC 2018 沈阳赛区网络预赛 J. Ka Chang  (分块思想)
		
题目链接:https://nanti.jisuanke.com/t/31451 题意: 给你一颗树,树上各点有初始权值,你有两种操作: 1. 给树中深度为l的点全部+x,(根节点为1,深度为0) 2. ...
 - ACM-ICPC 2018 沈阳赛区网络预赛 J Ka Chang
		
Ka Chang 思路: dfs序+树状数组+分块 先dfs处理好每个节点的时间戳 对于每一层,如果这一层的节点数小于sqrt(n),那么直接按照时间戳在树状数组上更新 如果这一层节点个数大于sqrt ...
 - ACM-ICPC 2018 沈阳赛区网络预赛 J. Ka Chang(树状数组+分块)
		
Given a rooted tree ( the root is node 1 ) of N nodes. Initially, each node has zero point. Then, yo ...
 - ACM-ICPC 2018 沈阳赛区网络预赛  J. Ka Chang (树分块)
		
题意:一个树,支持两种操作:1.将深度为L的节点权置加上X;2.求以x为根节点的子树上节点权置之和.根节点深度为0. 分析:考虑用树状数组维护节点权置,按dfs序下标查询.记录每个深度节点的个数.如果 ...
 - 【2018沈阳赛区网络预选赛J题】Ka Chang【分块+DFS序+线段树】
		
题意 给出一个有根树(根是1),有n个结点.初始的时候每个结点的值都是0.下面有q个操作,操作有两种,操作1.将深度为L的点的值全部增加X.操作2.查询以x为根的子树的结点值得和. 其中N,Q< ...
 - ACM-ICPC 2018 沈阳赛区网络预赛 J树分块
		
J. Ka Chang Given a rooted tree ( the root is node 11 ) of NN nodes. Initially, each node has zero p ...
 - ACM-ICPC 2018 南京赛区网络预赛 G  Lpl and Energy-saving Lamps(模拟+线段树)
		
https://nanti.jisuanke.com/t/30996 题意 每天增加m个灯泡,n个房间,能一次性换就换,模拟换灯泡过程.询问第几天的状态 分析 离线做,按题意模拟.比赛时线段树写挫了. ...
 - ACM-ICPC 2018 徐州赛区网络预赛  J. Maze Designer (最大生成树+LCA求节点距离)
		
ACM-ICPC 2018 徐州赛区网络预赛 J. Maze Designer J. Maze Designer After the long vacation, the maze designer ...
 - ACM-ICPC 2018 沈阳赛区网络预赛 K  Supreme Number(规律)
		
https://nanti.jisuanke.com/t/31452 题意 给出一个n (2 ≤ N ≤ 10100 ),找到最接近且小于n的一个数,这个数需要满足每位上的数字构成的集合的每个非空子集 ...
 
随机推荐
- MySQL基础之数据管理【5】
			
子查询的使用 select 字段名称 from tbl_name where col_name=(select col_name from tbl_name); --内层语句查询的结果可以作为外层语句 ...
 - SQL,case ziduan when ziduan_value then 'result'
			
case a.sex when 0 then '女' when 1 then '男' else '其他' end as sex 当a表的性别字段的value为0时将查询的value转换成 '女',当字 ...
 - [20191113]oracle共享连接模式端口2.txt
			
[20191113]oracle共享连接模式端口2.txt --//昨天的测试链接:http://blog.itpub.net/267265/viewspace-2663651/=>[20191 ...
 - SQL Server通过定义函数返回字段数据列表模板-干货
			
CREATE FUNCTION [dbo].[GetReportDWCustomerOrder] ( @YearDate DATETIME, 参数条件..... @Categor ...
 - Shell—常见报错问题
			
bash:$'\r': command not found 造成这个问题的原因是Windows环境下换行的“\r”到了Linux环境下不能够识别了,因为Linux环境下默认的换行符为“\n”,我们只需 ...
 - The 2017 ACM-ICPC Asia Beijing Regional Contest
			
传送门 C - Graph 题意: 给出一个\(n\)个点\(m\)条边的无向图.现在有多组询问,每组询问给出区间\([l,r]\),问区间\([l,r]\)中有多少点对是连通的. 思路: 若考虑只有 ...
 - 关于jvm的常识介绍
			
1.关于jvm的组成(只介绍1.8,以前的版本不多介绍) vm stack,native method stack,program counter register,heap,metaspace 2. ...
 - 阿里Nacos初体验
			
Nacos 从开源到现在已经18个releases了,更新很快,社区也很活跃,光钉钉群的人数那也是好多,每次点个查看全部成员的按钮,我的钉钉就要卡死,也有可能我没更新最新的版本. 前面写了一段时间的S ...
 - Linux 部署 nginx
			
nginx搭建 1. 清除之前nginx环境 #查看nginx进程 ps -ef|grep nginx #找到nginx相对应的位置 whereis nginx #停止nginx服务 /usr/loc ...
 - 网络聊天室---node.js中net网络模块TCP服务端与客户端的使用
			
//1.简单创建 net服务器 // const net = require("net"); // const server = net.createServer((c)=> ...