zoukankan      html  css  js  c++  java
  • 洛谷 4216 BZOJ 4448 [SCOI2015]情报传递

    【题解】

      每个情报员的危险值val[i]应该是一个分段函数,前面一段是平行于x轴的横线,后面一段是一次函数。我们可以用fx(t)=t-b[x]表示这个一次函数。每次询问一条链上fx(t)大于c的点的个数,也就是问有多少个点满足t-b[x]>c,移项得b[x]<t-c,不等式左边只与点有关,可以当做点权,右边只与询问有关。因此我们可以写没有修改的主席树。

      同时这道题也可以离线后用树状数组写。我们维护某个点到根的链上小于等于某个值的数的个数Cnt,这开一个权值树状数组就可以做到。这样每个询问(x,y)的答案就是Cnt[x]+Cnt[y]-Cnt[lca]-Cnt[fa[lca]]. 我们dfs进行维护,进入这个点的时候在树状数组中加入点权,退出这个点的时候删除点权,同时在每个点计算与这个点有关的答案即可。

      

     1 #include<cstdio>
     2 #include<algorithm>
     3 #include<queue>
     4 #define N 200010
     5 #define rg register
     6 using namespace std;
     7 int n,m,rt,tot,cnt,ret,last[N],ans[N],ans2[N],fa[N],hvy[N],size[N],top[N],dep[N],t[N],val[N];
     8 vector<int>son[N];
     9 struct rec{
    10     int pos,val,type,pre;
    11 }data[N<<2];
    12 inline int read(){
    13     int k=0,f=1; char c=getchar();
    14     while(c<'0'||c>'9')c=getchar();
    15     while('0'<=c&&c<='9')k=k*10+c-'0',c=getchar();
    16     return k*f;
    17 }
    18 void dfs1(int x){
    19     size[x]=1; dep[x]=dep[fa[x]]+1;
    20     for(rg int i=0,s;i<son[x].size();i++){
    21         dfs1(s=son[x][i]); size[x]+=size[s];
    22         if(size[s]>size[hvy[x]]) hvy[x]=s;
    23     }
    24 }
    25 void dfs2(int x,int tp){
    26     top[x]=tp;
    27     if(hvy[x]) dfs2(hvy[x],tp);
    28     for(rg int i=0,s;i<son[x].size();i++)
    29         if((s=son[x][i])!=hvy[x]) dfs2(s,s);    
    30 }
    31 inline int lca(int x,int y){
    32     int f1=top[x],f2=top[y];
    33     while(f1!=f2){
    34         if(dep[f1]<dep[f2]) swap(x,y),swap(f1,f2);
    35         x=fa[f1]; f1=top[x];
    36     }
    37     return dep[x]<dep[y]?x:y;
    38 }
    39 void dfs(int x){
    40     for(rg int i=val[x];i<=m;i+=(i&-i)) t[i]++;
    41     for(rg int i=last[x];i;i=data[i].pre){
    42         ret=0;
    43         for(rg int j=data[i].val-1;j>0;j-=(j&-j)) ret+=t[j];
    44         ans[data[i].pos]+=data[i].type*ret;
    45     }
    46     for(rg int i=0;i<son[x].size();i++) dfs(son[x][i]);
    47     for(rg int i=val[x];i<=m;i+=(i&-i)) t[i]--;
    48 }
    49 int main(){
    50     n=read();
    51     for(rg int i=1;i<=n;i++){
    52         fa[i]=read();
    53         if(!fa[i]) rt=i;
    54         son[fa[i]].push_back(i);
    55     }
    56     m=read();
    57     for(rg int i=1;i<=n;i++) val[i]=m;
    58     dfs1(rt); dfs2(rt,rt);
    59     for(rg int t=1;t<=m;t++){
    60         int opt=read();
    61         if(opt==1){
    62             int x=read(),y=read(),c=read(),Lca=lca(x,y);
    63             data[++tot]=(rec){++cnt,t-c,1,last[x]}; last[x]=tot;
    64             data[++tot]=(rec){cnt,t-c,1,last[y]}; last[y]=tot;
    65             data[++tot]=(rec){cnt,t-c,-1,last[Lca]}; last[Lca]=tot;
    66             data[++tot]=(rec){cnt,t-c,-1,last[fa[Lca]]}; last[fa[Lca]]=tot; 
    67             ans2[cnt]=dep[x]+dep[y]-2*dep[Lca]+1;
    68         }
    69         else{
    70             int x=read(); if(val[x]==m) val[x]=t;
    71         }
    72     }
    73     dfs(rt);
    74     for(rg int i=1;i<=cnt;i++) printf("%d %d
    ",ans2[i],ans[i]);
    75     return 0;
    76 }
    View Code
  • 相关阅读:
    微服务实战(二):使用API Gateway
    微服务实战(一):微服务架构的优势与不足
    在WIN7、WIN10操作系统用WebDAV映射网络驱动器需要的操作
    docker开机启动和docker-compose开机启动执行相应的各个docker容器
    /etc/rc.d/init.d自启动程序说明
    C# 通过反射实现对象映射:将2个属性相近的对象相互转换
    添加windows右键菜单:使用exe应用程序打开文件/文件夹
    .NET5 MVC Program.cs 笔记
    前端 JS 正则表达式积累
    VS Code 快捷键
  • 原文地址:https://www.cnblogs.com/DriverLao/p/8696630.html
Copyright © 2011-2022 走看看