• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            Tim's Programming Space  
            Tim's Programming Space
            日歷
            <2010年1月>
            272829303112
            3456789
            10111213141516
            17181920212223
            24252627282930
            31123456
            統計
            • 隨筆 - 20
            • 文章 - 1
            • 評論 - 40
            • 引用 - 0

            導航

            常用鏈接

            留言簿(3)

            隨筆檔案

            文章檔案

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

             
            聽說有版權問題不能貼題目?。那就只能先忍一忍了。
            題目抽象為:我們有一個由有根樹構成的森林,對這個森林進行兩種操作:
            把某棵子樹拔下來接到某一棵樹(可能還是那個子樹原來所在的樹)的某個節點下面,詢問某個節點在樹中的深度。

            因為把一棵邊權為1的樹的括號序列拿出來,樹上某兩點的距離就是在括號序列中兩點間沒匹配括號的個數(有左括號右括號選擇的區別,具體分析處理)。當然,既然是對一群樹操作,那就直接用動態樹就行了。

            于是就去學了動態樹。發現其實不算很難(1.指時間復雜度均攤logn的算法,還有基于輕重邊剖分的嚴格logn的算法 2.如果你對splay熟的話),寫起來也就基本上就是一棵splay,也算比較好寫的。。(以后就告別路徑剖分了。。太麻煩了。。復雜度也沒動態樹好。。)

            以下所說的動態樹都是基于splay的時間復雜度均攤logn的動態樹。

            動態樹的主要思想就是:類似輕重邊剖分一樣,把整棵樹劃分成若干實邊(solid edge)和虛邊(dashed edge),但這個都是根據你的需要來設定的,不像輕重邊一樣每個點往下都必須有一條重邊(單獨的葉子節點算長度為0的重邊),而是每次把你所需要操作的點到根的邊都改為實邊(expose操作),且每個點往下的實邊數不超過1。修改沿途如果有一個點已經有了實邊邊那么就把它原來的實邊改成虛邊。這樣每次對一個點操作都是在一條實路徑上(solid path)。對于每一條實路徑,都用一棵splay來維護就行了。(splay可以亂轉亂拔亂接太爽了。。- -!當然是在一定規則下的亂。。)


            /*
             * $File: bounce.cpp
             * $Date: Fri Jul 09 20:59:27 2010 +0800
             * $Author: Tim
             * $Solution: Dynamic Tree with Splay Tree implementation
             * $Time complexity: O(mlogn) , per operation amorized O(logn);
             
            */

            #include 
            <cstdio>
            #include 
            <cstdlib>
            #include 
            <cstring>
            #include 
            <cassert>

            #define MAXN 200005

            using namespace std;


            class SplayNode
            {
                
            public:
                    
            int fa, lt, rt, size;
            };
            SplayNode node[MAXN 
            + 1];

            // functions below are belong to splay tree
            // we can see that, this splay tree is quite
            // simple, and just 'splay' function
            // and size maintaining supported.
            // but that what all we need to
            // solve this problem

            void Renew(int x)
            {
                
            if (!x)
                    
            return;
                node[x].size 
            = node[node[x].lt].size + node[node[x].rt].size + 1;
            }
            void RightRotate(int x)
            {
                
            int lc = node[x].lt, fa = node[x].fa;
                node[x].lt 
            = node[lc].rt; node[node[x].lt].fa = x;
                node[lc].rt 
            = x; node[x].fa = lc;
                node[lc].fa 
            = fa;
                
            if (x == node[fa].lt)
                    node[fa].lt 
            = lc;
                
            else
                    node[fa].rt 
            = lc;
                Renew(x);
                Renew(lc);
            }


            void LeftRotate(int x)
            {
                
            int rc = node[x].rt, fa = node[x].fa;
                node[x].rt 
            = node[rc].lt; node[node[x].rt].fa = x;
                node[rc].lt 
            = x; node[x].fa = rc;
                node[rc].fa 
            = fa;
                
            if (x == node[fa].lt)
                    node[fa].lt 
            = rc;
                
            else
                    node[fa].rt 
            = rc;
                Renew(x);
                Renew(rc);
            }

            void splay(int x, int FA = 0)
            {
                
            int fa, Fa;
                
            while ((fa = node[x].fa) != FA)
                {
                    
            if ((Fa = node[fa].fa) == FA)
                    {
                        
            if (x == node[fa].lt)
                            RightRotate(fa);
                        
            else
                            LeftRotate(fa);
                    }
                    
            else
                    {
                        
            if (x == node[fa].lt)
                        {
                            
            if (fa == node[Fa].lt)
                            {
                                RightRotate(Fa);
                                RightRotate(fa);
                            }
                            
            else
                            {
                                RightRotate(fa);
                                LeftRotate(Fa);
                            }
                        }
                        
            else
                        {
                            
            if (fa == node[Fa].rt)
                            {
                                LeftRotate(Fa);
                                LeftRotate(fa);
                            }
                            
            else
                            {
                                LeftRotate(fa);
                                RightRotate(Fa);
                            }
                        }
                    }
                }
            }

            // end splay

            int root;
            int query_rank(int id)
            {
                splay(id);
                
            return node[node[id].lt].size + 1;
            }
            int father[MAXN + 1];
            int n;
            void Init()
            {
                scanf(
            "%d"&n);
                
            for (int i = 1, k; i <= n; i ++)
                {
                    scanf(
            "%d"&k);
                    k 
            += i;
                    
            if (k > n + 1)
                        k 
            = n + 1;
                    father[i] 
            = k;
                    node[i].size 
            = 1;
                }
                node[n 
            + 1].size = 1;
            }

            int split(int id) 
                
            // isolate id and the node right after it on the solid path 
                
            // and return that node
            {
                splay(id);
                
            if (node[id].rt)
                {
                    
            int rc = node[id].rt;
                    node[id].rt 
            = node[rc].fa = 0;
                    node[id].size 
            -= node[rc].size;
                    
            return rc;
                }
                
            else
                    
            return 0;
            }

            void Link(int id, int fa) 
                
            // let fa be the father of id, 
                
            // we assume that before this, 
                
            // id is the head of a solid path,
                
            // and fa is the tail of a solid path,
                
            // this was done by function 'cut' and 'split'
            {
                splay(id);
                assert(
            !node[id].lt);
                splay(fa);
                assert(
            !node[fa].rt);
                node[fa].rt 
            = id;
                node[fa].size 
            += node[id].size;
                node[id].fa 
            = fa;
            }

            int get_head(int x)
                
            // get the head of the solid path which x is in.
            {
                
            while (node[x].fa)
                    x 
            = node[x].fa;
                
            while (node[x].lt)
                    x 
            = node[x].lt;
                splay(x);
                
            return x;
            }

            void expose(int id) 
                
            // turn the edges between id and the root of the tree id is in
                
            // all into solid edges. with this operation, we can query what
                
            // we want conveniently in a splay tree.
            {
                
            while (true)
                {
                    id 
            = get_head(id);
                    
            if (!father[id])
                        
            break;
                    split(father[id]);
                    Link(id, father[id]);
                }
            }

            int query_depth(int id)
            {
                expose(id);
                
            return query_rank(id) - 1;
            }

            void cut(int id)
                
            // this function isolated the subtree rooted id
            {
                expose(id);
                split(father[id]);
            }

            void modify_father(int id, int fa)
            {
                cut(id);
                split(fa);
                father[id] 
            = fa;
                Link(id, fa);
            }

            void Solve()
            {
                
            int m, cmd, id, k;
                scanf(
            "%d"&m);
                
            while (m --)
                {
                    scanf(
            "%d%d"&cmd, &id);
                    id 
            ++;
                    
            if (cmd == 1)
                        printf(
            "%d\n", query_depth(id));
                    
            else
                    {
                        scanf(
            "%d"&k);
                        k 
            += id;
                        
            if (k > n + 1)
                            k 
            = n + 1;
                        modify_father(id, k);
                    }
                }
            }

            int main()
            {
                freopen(
            "bounce.in""r", stdin);
                freopen(
            "bounce.out""w", stdout);
                Init();
                Solve();
                
            return 0;
            }


            posted on 2010-07-09 21:00 TimTopCoder 閱讀(3196) 評論(5)  編輯 收藏 引用
            評論:
            • # re: HNOI2010-彈飛綿羊-(splay維護括號序列 或 動態樹)  treeboy Posted @ 2010-07-10 08:53
              弱弱地問下:怎么把文章放在首頁,我是新用戶  回復  更多評論   

            • # re: HNOI2010-彈飛綿羊-(splay維護括號序列 或 動態樹)  zxytim Posted @ 2010-07-10 09:49
              @treeboy
              發表的時候選擇 首頁精華區。。
              但你要保證你的文章大家看了會受益。。  回復  更多評論   

            • # re: HNOI2010-彈飛綿羊-(splay維護括號序列 或 動態樹)[未登錄]  treeboy Posted @ 2010-07-10 14:21
              @zxytim
              博主,我的意思是把文章放在自己的首頁,如我的www.shnenglu.com/treeboy
              謝謝  回復  更多評論   

            • # re: HNOI2010-彈飛綿羊-(splay維護括號序列 或 動態樹)  zxytim Posted @ 2010-07-10 16:20
              哦。。。
              點 發新隨筆 不要點 發新文章。。。  回復  更多評論   

            • # re: HNOI2010-彈飛綿羊-(splay維護括號序列 或 動態樹)  cnx Posted @ 2010-07-27 17:17
              求數據!bobchennan@163.com  回復  更多評論   

             
            Copyright © TimTopCoder Powered by: 博客園 模板提供:滬江博客
            久久精品国产99久久无毒不卡| 国内精品伊人久久久影院| 国产精品久久午夜夜伦鲁鲁| 国产精品99精品久久免费| 亚洲国产精久久久久久久| 久久综合狠狠综合久久97色| 久久综合给合久久狠狠狠97色| 91麻豆精品国产91久久久久久| 伊人 久久 精品| 亚洲乱亚洲乱淫久久| 亚洲精品蜜桃久久久久久| 久久人人爽人人爽人人片AV麻豆| 色偷偷久久一区二区三区| 四虎影视久久久免费| 69久久夜色精品国产69| 欧美黑人激情性久久| 韩国三级中文字幕hd久久精品 | 国産精品久久久久久久| 久久99热这里只有精品国产| 国内精品久久久久国产盗摄| 色欲综合久久中文字幕网| 国产精品久久久久久久久| 思思久久99热免费精品6| 久久综合久久综合久久| 蜜臀av性久久久久蜜臀aⅴ麻豆 | 久久成人18免费网站| 97久久香蕉国产线看观看| 伊人色综合久久天天人手人婷| 久久99精品久久久久久9蜜桃| 国产精品久久久久久久久免费| 亚洲乱码精品久久久久..| 精品国产乱码久久久久久呢| 久久精品国产男包| 久久婷婷五月综合色奶水99啪| 精品多毛少妇人妻AV免费久久| 国产精品青草久久久久福利99| 国产精品久久亚洲不卡动漫| 久久精品九九亚洲精品天堂| 秋霞久久国产精品电影院| 品成人欧美大片久久国产欧美| 亚洲狠狠久久综合一区77777 |