zoukankan      html  css  js  c++  java
  • 数据结构练习

    P2320 - 【长郡模拟赛】数据结构练习

    Description

    费了一番功夫,神犇 CJK 终于完成了前三道题目。“不错,不愧是新一代神犇 啊!” JesseLiu 满意地说道,“不过,你在算法方面的功底固然不错。对于数据结 构的运用,你又掌握地如何呢?”
    听到“数据结构”这四个字,就连身为神犇的 CJK 也不禁吓出一身冷汗。“年轻 人,现在,对于我给定一棵树,你需要完成以下操作:
    1.修改某个点的权值;
    2.查询某两点间路径上所有点的权值和;
    3.查询某点子树的权值和。”
    CJK 脸上闪过一丝不屑:不就是道链剖裸题吗?
    “这只是第一问。”JesseLiu 似乎也觉得这道题太水了,于是补充道:“完成以 上所有操作后,我还会有以下几种询问:
    1. 询问某个点子树上有多少个点的权值小于等于 k;
    2. 询问某两点之间的路径上有多少点的权值小于等于 k;”
    尽管 CJK 是神犇,但遇到这种题,也不禁感到一丝恐惧。还好,通过自己的 玄学力量,他联系到了他的同学——你。现在,就请你 A 掉这最后一道水题。

    Input

    第一行一个数 n,表示点的总数。接下来 n-1 行,一行两个整数 x 和 y,表示 x 和 y 之间有边相连。接下来一个整数 m1,表示第一问中操作的数量。接下来 m1 行表示操作,格式如下:
    “1 x y”将点 x 的权值修改为 y;
    “2 x y”询问 x 到 y 路径上的点权之和(包括 x 和 y);
    “3 x”询问 x 的子树上的点权和(包括 x);
    接下来一行 m2,表示第二问中操作的数量。接下来 m2 行描述操作,格式 如下:
    “1 x y z”询问 x 到 y 的路径上有多少个点的权值小于等于 z;
    “2 x y”询问 x 的子树上有多少个点的权值小于等于 y;
    每个点的初始权值为 0,默认树根为 1 号节点。

    Output

    对于每一次询问,输出一个整数表示询问的答案。

    Sample Input

    样例①:
    3
    1 2
    1 3
    3
    1 1 3
    1 3 1
    3 1
    2
    2 1 2
    1 1 3 1

    样例②:
    5
    1 2
    1 3
    3 4
    3 5
    5
    3 1
    1 3 1
    2 4 5
    1 4 -1
    3 3
    2
    1 1 5 0
    2 3 -1

    Sample Output

    样例①:
    4
    2
    1

    样例②:
    0
    1
    0
    2
    1

    Hint

    对于 30%的数据,n<=100,m1<=100,m2<=100。剩下的数据保证 n<=50000;
    对于另外 30%的数据,m2=0。其中有 1 组数据,树退化为链;
    对于 100%的数据, 保证 n<=50000,m1<=50000,m2<=50000, 保证题目中所有输入、 输出、中间运算不超过 int 范围。

    Source

    叶闻捷出题 NOIP 模拟测试T4

     

    第二问:离线做法。

           我们可以与第一问类比一下:对于第一问,所有对答案有贡献的点只有一个约束条件:该点在给定两点的路径上(或在给定点的子树上)。而一个点对答案的贡献就是它的权值,所以用树链剖分区间求和即可。

           再看看第二问:一个点要对答案有贡献,必须满足两个条件:(1)该点在给定两点的路径上(或在给定点的子树上);(2)该点的权值小于等于一个给定的值k。而每个点对于答案的贡献是1。

           如果只要求满足第一个条件,那么我们只需要像第一问一样区间求和就可以了。但是,还有第二个条件的约束。所以,我们需要保证:在询问时,只有权值小于等于k(k的意义见上)的点能对答案产生贡献。也就是说,只有权值小于等于k的点对应的dfs序在线段树中对应的点为1,其它都是0

           所以,我们可以把树上的每一个点当做一次修改操作(把对应的dfs序在线段树中对应的值修改为1),每一个询问操作就是统计路径(子树)上有多少个1(其实就是一次区间求和,和第一问一模一样)。我们只需要在进行操作之前,对于所有的操作,以权值为关键字(修改操作的权值是对应的点权,询问操作的权值为对应的k)由小到大排序,权值相同的把修改放到查询之前。然后和第一问一样地跑就可以了。
     
      1 #include<algorithm>
      2 #include<iostream>
      3 #include<iomanip>
      4 #include<cstring>
      5 #include<cstdlib>
      6 #include<cstdio>
      7 #include<queue>
      8 #include<ctime>
      9 #include<cmath>
     10 #include<stack>
     11 #include<map>
     12 #include<set>
     13 #define rep(i,a,b) for(register int i=a;i<=b;i++)
     14 #define il inline
     15 #define ll long long
     16 #define RG register
     17 #define ls o<<1
     18 #define rs (o<<1)|1
     19 #define mimi RG int mid=(L+R)>>1;
     20 using namespace std;
     21 const int N=50010,M=50010;
     22 struct E {
     23   int to,net;
     24 }e[N*2];
     25 struct ASK{
     26   int x,y,k;
     27   int f;
     28 }a[M],dig[M];
     29 int head[N],top[N],tid[N],pos[N],siz[N],dep[N],son[N],fa[N],oh[N],idx,num_e,n,m1,m2;
     30 int W[N],sum[N];
     31 int gi();
     32 void add(int x,int y) {
     33   e[++num_e].to=y , e[num_e].net=head[x] , head[x] = num_e;
     34 }
     35 int comp(const ASK& a,const ASK& b) {
     36   if(a.k<b.k) return 1;
     37   if(a.k>b.k) return 0;
     38   return a.f>b.f;
     39 }
     40 void dfs1(int x,int fu) {
     41   siz[x]=1;
     42   dep[x]=dep[fu]+1;
     43   fa[x]=fu;
     44   for(RG int i=head[x];i;i=e[i].net) {
     45     int to=e[i].to;
     46     if(to==fu) continue;
     47     dfs1(to,x);
     48     siz[x] += siz[to];
     49     if(siz[to]>siz[son[x]]) son[x]=to;
     50   }
     51 }
     52 void dfs2(int x,int tp) {
     53   tid[x]=++idx;
     54   pos[idx]=x;
     55   top[x]=tp;
     56   oh[x]=idx;
     57   if(!son[x]) return;
     58   dfs2(son[x],tp);// bug 
     59   oh[x]=oh[son[x]];
     60   for(RG int i=head[x];i;i=e[i].net) {
     61     int to=e[i].to;
     62     if(to!=fa[x]&&to!=son[x]) dfs2(to,to),oh[x]=max(oh[x],oh[to]);
     63   }
     64 }
     65 void Update(int o,int L,int R,int p,int x) {
     66   if(L==R&&p==L) {
     67     sum[o]=x;return;
     68   }
     69   mimi;
     70   if(p<=mid) Update(ls,L,mid,p,x);
     71   else Update(rs,mid+1,R,p,x);
     72   sum[o]=sum[ls]+sum[rs];
     73 }
     74 int querysum(int o,int L,int R,int l,int r) {
     75   if(l<=L&&R<=r) return sum[o];
     76   mimi;
     77   int he=0;
     78   if(l<=mid) he += querysum(ls,L,mid,l,r);
     79   if(r>mid) he += querysum(rs,mid+1,R,l,r);
     80   return he;
     81 }
     82 int solve(int x,int y) {
     83   int he=0;
     84   //  rep(i,1,n) printf("top=%d ",top[i]);puts("
    ");
     85   while(top[x]!=top[y]) {
     86     if(dep[top[x]]>dep[top[y]]) swap(x,y);
     87     he += querysum(1,1,n,tid[top[y]],tid[y]);
     88     y=fa[top[y]];
     89   }
     90   if(dep[x]>dep[y]) swap(x,y);
     91   he += querysum(1,1,n,tid[x],tid[y]);
     92   return he;
     93 }
     94 void ONE() {
     95   n=gi();
     96   RG int x,y;
     97   rep(i,1,n-1) {
     98     x=gi(),y=gi();add(x,y),add(y,x);
     99   }
    100   dfs1(1,0);
    101   dfs2(1,1);
    102   m1=gi();RG int k;
    103   while(m1--) {
    104     k=gi(),x=gi();
    105     if(k==1) y=gi(),Update(1,1,n,tid[x],y),W[tid[x]]=y;// val[x]
    106     else if(k==2) y=gi(),printf("%d
    ",solve(x,y));
    107     else printf("%d
    ",querysum(1,1,n,tid[x],oh[x]));
    108   }
    109   return;
    110 }
    111 void TWO() {
    112   m2=gi();
    113   memset(sum,0,sizeof(sum));
    114   RG int x,y,k;// ghost
    115   rep(i,1,m2) {
    116     k=gi(),x=gi(),y=gi();
    117     if(k==1) a[i].x=x,a[i].y=y,a[i].k=gi();
    118     else a[i].x=x,a[i].y=pos[oh[x]],a[i].k=y;
    119   }
    120   int num=m2;
    121   rep(i,1,n) a[++num].x=i,a[num].k=W[tid[i]],a[num].f=1;
    122   sort(a+1,a+num+1,comp);
    123   rep(i,1,num) printf("%d %d %d
    ",a[i].x,a[i].y,a[i].f);
    124   rep(i,1,num) {
    125     if(a[i].f==0) printf("%d
    ",solve(a[i].x,a[i].y));
    126     else Update(1,1,n,tid[a[i].x],1);
    127   }
    128   return;
    129 }
    130 int main() {
    131     freopen("datastruct.in","r",stdin);
    132     freopen("datastruct.out","w",stdout);
    133     ONE();
    134     TWO();
    135     return 0;
    136 }
    137 int gi() {
    138     int res=0,f=1;
    139     char ch=getchar();
    140     while((ch<'0'||ch>'9')&&ch!='-') ch=getchar();
    141     if(ch=='-') ch=getchar(),f=-1;
    142     while(ch>='0'&&ch<='9') res=res*10+ch-'0',ch=getchar();
    143     return res*f;
    144 }
  • 相关阅读:
    流行-Manifold学习理解与应用
    狠心奶奶自断亲情,28年后孙女拒绝相见:人有没有不原谅的权利?
    学术论文常用词汇总结(待更新)
    机动车驾驶(2)--- 老司机经验
    关于MySQL数据导出导入
    php5.6-lumen与php5.6-phalcon性能对比
    win7(64bit)+python3.5+pyinstaller3.2安装和测试
    WARNING: Can not get binary dependencies for file...
    一些不错的计算机书籍
    PHP扩展类ZipArchive实现压缩解压Zip文件和文件打包下载
  • 原文地址:https://www.cnblogs.com/ypz999/p/6674936.html
Copyright © 2011-2022 走看看