zoukankan      html  css  js  c++  java
  • [jzoj]5257.小X的佛光

    Link

      https://jzoj.net/senior/#main/show/5257

    Problem

    Solution

    5~90分

      我们可以根据特殊性质搞

      如果数据小,直接暴力在树上面模拟一次

      如果满足性质1,就是第i条边连i和i+1地,那么就成了一条链,答案是可以按照数学方法计算出来的

      如果满足性质2,就是A=C,那么就是A~B的长度,可以预处理快速求出来

      这些分数,感谢出题人的馈赠

    100分

      其实我们可以画个图,粗略的画个图,看看答案是什么

      

      显然,根据这个图,我们就知道,答案其实就是BX这条线段的长度

      怎么求?我们可以用最简单的容斥原理

      

      加一是因为X这个点之后要div2,所以应该得被计算2次

      我们发现,求AB,CB,AC的方法,都是求他们到lca的长度

      所以,我们可以用倍增,对他们做lca,并且记录路径长度

      我们一次递归,预处理出,每个点的深度。

      在递归中,顺便预处理f数组

      其中f[i,j]表示第i个节点,在他上面第2j个节点是什么。

      我们先把深度大的点,用倍增,弄成和另外一个节点一样的深度,这个和下面的方法类似

      然后同时倍增。

      其实就是找一个最大的j,满足f[a,j]不等于f[b,j],一直弄,最后就成了lca的两个儿子。

      加法随便弄一下就行了。

      其他两个求法类似,不在赘述了。

      时间复杂度:O(3n)(n表示点数)

    Code

    {$inline on}
    var
            n,m,i,j,x,y,z,tot:longint;
            f:array[0..200000,0..17] of longint;
            pre,l,d:array[0..400000] of longint;
            dis,shen:array[0..200000] of longint;
    procedure insert(x,y:longint); inline;
    begin
            inc(tot);
            d[tot]:=y;
            pre[tot]:=l[x];
            l[x]:=tot;
    end;
    
    procedure dg(now,k,q:longint);  inline;
    var
            s:longint;
    begin
            shen[now]:=k;
            f[now,0]:=q;
            s:=l[now];
            while s<>0 do
            begin
                    if dis[d[s]]=0 then
                    begin
                            dis[d[s]]:=1;
                            dg(d[s],k+1,now);
                    end;
    
                    s:=pre[s];
            end;
    end;
    
    function yes(x,y:longint):longint; inline;
    var
            ans,k:longint;
    begin
            ans:=1;
            if shen[x]>shen[y] then
            begin
                    while shen[x]>shen[y] do
                    begin
                            for k:=17 downto 0 do
                                    if shen[f[x,k]]>=shen[y] then
                                            break;
    
                            x:=f[x,k];
                            ans:=ans+1 shl k;
                    end;
            end;
    
            if shen[x]<shen[y] then
            begin
                    while shen[x]<shen[y] do
                    begin
                            for k:=17 downto 0 do
                                    if shen[f[y,k]]>=shen[x] then
                                            break;
    
                            y:=f[y,k];
                            ans:=ans+1 shl k;
                    end;
            end;
    
            while x<>y do
            begin
                    for k:=17 downto 0 do
                            if (f[x,k]<>f[y,k]) and (f[x,k]<>0) and (f[y,k]<>0) then
                                    break;
    
                    x:=f[x,k];
                    y:=f[y,k];
                    ans:=ans+1 shl (k+1);
            end;
    
            exit(ans);
    end;
    
    begin
            readln(n,m,x);
            for i:=1 to n-1 do
            begin
                    readln(x,y);
    
                    insert(x,y);
                    insert(y,x);
            end;
    
            dis[1]:=1;
            shen[0]:=-maxlongint;
            dg(1,1,0);
    
            for j:=1 to 17 do
                    for i:=1 to n do
                            f[i,j]:=f[f[i,j-1],j-1];
    
            while m>0 do
            begin
                    readln(x,y,z);
    
                    writeln((yes(x,y)+yes(y,z)-yes(x,z)+1) shr 1);
    
                    dec(m);
            end;
    end.
  • 相关阅读:
    JavaScript 命名空间
    雅虎网站页面性能优化的34条黄金守则
    利用模板引擎配合ajax进行数据的导入
    canvas 实现小人的行走和上下左右的变换
    canvas 做一个小鸟运动的小游戏 (第二步) 使小鸟飞起来
    canvas 做一个小鸟运动的小游戏 (第一步)
    canvas 画一个小时钟
    更改博客的通知
    pat advanced 1139. First Contact (30)
    10分钟上手python pandas
  • 原文地址:https://www.cnblogs.com/philchieh/p/7365689.html
Copyright © 2011-2022 走看看