• <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>

            M.J的blog

            algorithm,ACM-ICPC
            隨筆 - 39, 文章 - 11, 評論 - 20, 引用 - 0
            數(shù)據(jù)加載中……

            TOJ 3446.Money Matters 并查集,路徑壓縮

                    題目大意是N個人互相有債a[i](正的表示別人欠錢,否則表示自己欠別人錢,a[i]的和保證為0),但是這N個人只有朋友間才能互相算帳,現(xiàn)在給出N個人的債,并且給出M個關(guān)系,即誰和誰是朋友,最后問有沒有可能所有人都把帳算清。


            Sample Input 1:                         Sample Input 2:

            5    3                                            4     2
            100                                              15
            -75                                               20
            -25                                              -10
            -42                                              -15
            42                                                0  2
            0 1                                               1  3
            1 2
            3 4

            Sample Output 1:                       Sample Output  2:

            POSSIBLE                                    IMPOSSILBE

             思路1:

                          每輸入一個關(guān)系,合并,最后從1到N,對于每一個人,找出它們的祖先,將他們的債debt加到祖先的debt

                          上。通俗的講,就是每個集合的成員都將自己的debt加到祖先上,自己歸0,最后看祖先的值是否為0;

                          最后在從1到N掃一遍如果有debt不為0的,輸出“IMPOSSIBLE”,否則“POSSIBLE”;

                          缺點:復(fù)雜度高,查詢復(fù)雜度*N,最后勉強過了,時間0.63s

            思路2:

                          路徑壓縮,每輸入一個關(guān)系,合并。最后從1到N,如果 i 的debt不為0,從 i 到 i 的祖先的路徑不斷壓縮直到

                          祖先,這樣雖然仍然是N次操作,但由于在壓縮過程中大部分成員debt歸0,故實際上進行操作的次數(shù)遠小于N.

                          最后時間為0.11,比起上一種方法有了很大提高

            Code 1:

            #include <cstdio>
            #include 
            <cstring>
            struct Node{
                    
            int father,v,num;
            }a[
            10002];
            int find(int n){
                    
            int tep,m = n;
                    
            while(n != a[n].father)
                            n 
            = a[n].father;
                    
            while(m != n){
                            tep 
            = a[n].father;
                            a[n].father 
            = n;
                            m 
            = tep;
                    }
                    
            return n;
            }
            void Union(int root1,int root2){
                    
            int t1,t2;
                    t1 
            = find(root1),t2 = find(root2);
                    
            if(t1 == t2) return ;
                    
            else{
                            
            if(a[t1].v > a[t2].v){
                                    a[t1].v 
            += a[t2].v;
                                    a[t2].father 
            = t1;
                            }
                            
            else{
                                    a[t2].v 
            += a[t1].v;
                                    a[t1].father 
            = t2;
                            }
                    }
            }
            int main()
            {
                    
            int n,m,i,j;
                    
            bool flag = false;
                    scanf(
            "%d%d",&n,&m);
                    
            for(i = 0;i < n; ++i){
                            a[i].father 
            = i,a[i].v = 0;
                            scanf(
            "%d",&a[i].num);
                    }
                    
            while(m--){
                            scanf(
            "%d%d",&i,&j);
                            Union(i,j);
                    }
                    
            for(i = 0;i < n; ++i){
                            
            int tep = find(i);
                            
            int tt = a[i].num;
                            a[i].num 
            -= tt;
                            a[tep].num 
            += tt;
                    }
                    
            for(i = 0;i < n; i++)
                            
            if(a[i].num != 0){
                                    flag 
            = true;
                                    
            break;
                            }
                    
            if(flag) printf("IMPOSSIBLE\n");
                    
            else     printf("POSSIBLE\n");
            }

            Code 2:

            #include <cstdio>
            #include 
            <cstring>
            struct Node{
                    
            int father,v,num;
            }a[
            10002];
            int find(int n){
                    
            int m = n,tep;
                    
            while(n != a[n].father)
                            n 
            = a[n].father;
                    
            while(m != n){
                            tep 
            = a[m].father;
                            
            int tt = a[m].num;
                            a[m].num 
            -= tt;
                            a[tep].num 
            += tt;
                            a[m].father 
            = n;
                            m 
            = tep;
                    }
                    
            return n;
            }
            void Union(int root1,int root2){
                    
            int t1,t2;
                    t1 
            = find(root1),t2 = find(root2);
                    
            if(t1 == t2) return ;
                    
            else{
                            
            if(a[t1].v > a[t2].v){
                                    a[t1].v 
            += a[t2].v;
                                    a[t2].father 
            = t1;
                            }
                            
            else{
                                    a[t2].v 
            += a[t1].v;
                                    a[t1].father 
            = t2;
                            }
                    }
            }
            int main()
            {
                    
            int n,m,i,j;
                    
            bool flag = false;
                    scanf(
            "%d%d",&n,&m);
                    
            for(i = 0;i < n; ++i){
                            a[i].father 
            = i,a[i].v = 0;
                            scanf(
            "%d",&a[i].num);
                    }
                    
            while(m--){
                            scanf(
            "%d%d",&i,&j);
                            Union(i,j);
                    }
                    
            for(i = 0;i < n; ++i)
                         
            if(a[i].num != 0) find(i);
                    
            for(i = 0;i < n; i++)
                            
            if(a[i].num != 0){
                                    flag 
            = true;
                                    
            break;
                            }
                    
            if(flag) printf("IMPOSSIBLE\n");
                    
            else     printf("POSSIBLE\n");
            }








            posted on 2010-07-05 21:08 M.J 閱讀(306) 評論(0)  編輯 收藏 引用


            只有注冊用戶登錄后才能發(fā)表評論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            国产高潮国产高潮久久久91 | 久久人人爽人人爽人人片av麻烦 | 国产精品99久久精品| 精品国产乱码久久久久久郑州公司 | 99久久免费国产精精品| 国产精品欧美久久久久无广告| 97超级碰碰碰碰久久久久| 久久综合狠狠综合久久97色| 亚洲AV乱码久久精品蜜桃| 久久国产一区二区| 麻豆av久久av盛宴av| 国产精品久久久久久久 | 久久天堂AV综合合色蜜桃网| 国产成人精品久久| 韩国免费A级毛片久久| 久久免费99精品国产自在现线 | 国内精品久久久久久久亚洲| 久久精品人成免费| 国产精品久久久久久五月尺| 国产精自产拍久久久久久蜜| 久久99精品久久只有精品 | 中文字幕乱码人妻无码久久| 久久影院亚洲一区| 97久久精品人人澡人人爽| 日产精品99久久久久久| 伊人色综合久久天天人守人婷| 精品久久久久久无码国产| 国产精品一区二区久久| 久久久一本精品99久久精品66| 久久人人爽人人爽人人片AV东京热| 久久99毛片免费观看不卡| 久久人妻少妇嫩草AV无码专区| 思思久久好好热精品国产| 欧美一级久久久久久久大| 国产香蕉97碰碰久久人人| 久久久久一区二区三区| 色综合色天天久久婷婷基地| 久久成人影院精品777| 久久精品国产免费| 久久精品成人免费国产片小草| 久久国产乱子伦精品免费午夜|