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

            coreBugZJ

            此 blog 已棄。

            ACM搜索題經典 Sticks

            /*
            EOJ  1981 Sticks
            POJ  1011 Sticks
            HDOJ 1455 Sticks
            UVA  307  Sticks


            ----問題描述:

            George took sticks of the same length and cut them randomly until all parts became at most 50 units long.
            Now he wants to return sticks to the original state, but he forgot how many sticks he had originally


            ----輸入:

            The input contains blocks of 2 lines.
            The first line contains the number of sticks parts after cutting, there are at most 64 sticks.
            The second line contains the lengths of those parts separated by the space. The last line of the file contains zero.


            ----輸出:

            The output should contains the smallest possible length of original sticks, one per line.


            ----樣例輸入:

            9
            5 2 1 5 2 1 5 2 1
            4
            1 2 3 4
            0


            ----樣例輸出:

            6
            5


            ----分析:

            從短到長枚舉原始木棍長度,然后嘗試能否拼裝成功,
            第一次嘗試成功時的原始木棍長度,即為答案。

            嘗試方法,為深度優先搜索,具體剪枝策略見代碼注釋。


            ----幾組比較強的測試數據:

            input

            64
            40 40 30 35 35 26 15 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 43 42 42 41 10 4 40 40 40 40 40 40 40 40 40 40 40 40 40 40 25 39 46 40 10 4 40 40 37 18 17 16 15 40 40 40 40 40 40 40 40
            64
            40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 43 42 42 41 10 4 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40 40
            64
            33 36 7 45 6 14 14 14 28 39 35 36 5 33 21 32 29 37 31 3 2 19 40 34 25 1 33 49 20 14 25 36 45 37 47 28 39 8 36 44 8 48 41 1 13 18 9 10 34 42 41 39 42 20 23 6 40 28 49 16 38 33 15 7

            output

            454
            1251
            81

            */




            /**********************************************************
            版本三:
            EOJ  1981   0MS  231K
            POJ  1011  16MS  164K
            HDOJ 1455   0MS  200K
            UVA  307   WA

            增加搜索次數限制。

            (UVA 上加此限制則 WA,否則 0.212 AC,見版本二)
            */


            #include 
            <stdio.h>
            #include 
            <stdlib.h>
            #include 
            <string.h>

            #define  L  70

            int cutnum;                 // 木棍的數量
            int cutlen[ L ], totlen;    // 木棍的各自長度,及總長度

            int cmp( const void *pa, const void *pb) {
                    
            return (*((int*)pb)) - (*((int*)pa));
            }


            int orglen, orgnum; // 原始木棍的長度及其數量
            int used[ L ];      // dfs 時標記某木棍是否已經被用于組裝原始木棍
            //int choice[ L ];

            // 若搜索次數超過此值,則認為不可能成功
            #define  LIM  100000
            int lim;
                    
            // 嘗試拼裝某原始木棍,
                    
            // 此次嘗試之前,
                    
            // 已經使用了 t 根木棍,
                    
            // 已經拼裝出了原始木棍的部分長度,剩余 c 長度需要拼裝,
                    
            // 此次嘗試,只需要從第 b 根木棍開始。
            int dfs(int c, int b, int t) {
                    
            int i, prelen = -1;
                    
            if ( --lim <= 0 ) {
                            
            return 0;
                    }

                    
            if ( 0 == c ) {
                            
            // 剩余長度為 0,則開始拼裝下一根原始木棍,
                            
            // 需擴大嘗試的范圍。
                            for ( i = 0; (i < cutnum) && (used[i]); ++i ) {
                            }

                            
            // 所有木棍都已經被用于拼裝原始木棍,
                            
            // 即嘗試成功。
                            if ( i >= cutnum ) {
                                    
            //printf("i >= cutnum, t = %d\n", t);
                                    return 1;
                            }

                            used[ i ] 
            = 1;
                            
            //choice[ t ] = i;
                            
            // 任一原始木棍中,必含有剩余木棍中最長的那根
                            if ( dfs(orglen-cutlen[i], i+1, t+1) ) {
                                    
            return 1;
                            }

                            used[ i ] 
            = 0;
                            
            return 0;
                    }

                    
            for ( i = b; i < cutnum; ++i ) {
                            
            if (    (used[ i ]) ||          // 木棍沒用過
                                    (c < cutlen[i]) ||      // 剩余長度能容納此木棍
                                    (prelen == cutlen[i])   // 此長度的木棍已經試過,不行,無需再試
                            ) {
                                    
            continue;
                            }

                            used[ i ] 
            = 1;
                            
            //choice[ t ] = i;

                            
            // 下次嘗試從后面的木棍開始,前面的木棍無需考慮
                            if ( dfs(c-cutlen[i], i+1, t+1) ) {
                                    
            return 1;
                            }


                            prelen 
            = cutlen[ i ];

                            used[ i ] 
            = 0;

                            
            // 存在一木棍剛好可以符合剩余長度,則不再嘗試其它
                            if ( c == cutlen[ i ] ) {
                                    
            return 0;
                            }

                    }

                    
            return 0;
            }


            int find(){
                    
            // 原始木棍長度必為總長度的約數
                    if ( 0 != totlen % orglen ) {
                            
            return 0;
                    }

                    orgnum 
            = totlen / orglen;
                    memset(used, 
            0sizeof(used));
                    lim 
            = LIM;
                    
            return dfs(000);
            }


            int main() {
                    
            int i;
                    
            while ( (1 == scanf("%d"&cutnum)) && (0 < cutnum) ) {
                            totlen 
            = 0;
                            
            for ( i = 0; i < cutnum; ++i ) {
                                    scanf(
            "%d", cutlen+i);
                                    totlen 
            += cutlen[ i ];
                            }

                            
            // 將木棍從長到短排序,然后從長到短搜索之
                            qsort(cutlen, cutnum, sizeof(cutlen[0]), cmp);

                            
            /*for ( i = 0; i < cutnum; ++i ) {
                                    printf("-%d-", cutlen[ i ]);
                            }
                            printf("\n");
            */


                            
            // 原始木棍長度必大于等于最長木棍長度,
                            
            // 小于等于木棍總長度,
                            
            // 從短到長枚舉原始木棍長度。
                            for ( orglen = cutlen[0]; (orglen < totlen) && (! find()); ++orglen ) {
                            }

                            printf(
            "%d\n", orglen);
                            
            //if ( orglen != totlen ) {
                            
            //        for ( i = 0; i < cutnum; ++i ) {
                            
            //                printf("--%d--", choice[ i ]);
                            
            //        }
                            
            //        printf("\n");
                            
            //}
                    }

                    
            return 0;
            }



            /**********************************************************
            版本二:
            EOJ  1981  TLE
            POJ  1011  0MS   164K
            HDOJ 1455  0MS   188K
            UVA  307   0.212
            */

            /*
            #include <stdio.h>
            #include <stdlib.h>
            #include <string.h>

            #define  L  70

            int cutnum;                 // 木棍的數量
            int cutlen[ L ], totlen;    // 木棍的各自長度,及總長度

            int cmp( const void *pa, const void *pb) {
                    return (*((int*)pb)) - (*((int*)pa));
            }

            int orglen, orgnum; // 原始木棍的長度及其數量
            int used[ L ];      // dfs 時標記某木棍是否已經被用于組裝原始木棍
            //int choice[ L ];

                    // 嘗試拼裝某原始木棍,
                    // 此次嘗試之前,
                    // 已經使用了 t 根木棍,
                    // 已經拼裝出了原始木棍的部分長度,剩余 c 長度需要拼裝,
                    // 此次嘗試,只需要從第 b 根木棍開始。
            int dfs(int c, int b, int t) {
                    int i, prelen = -1;
                    if ( 0 == c ) {
                            // 剩余長度為 0,則開始拼裝下一根原始木棍,
                            // 需擴大嘗試的范圍。
                            for ( i = 0; (i < cutnum) && (used[i]); ++i ) {
                            }
                            // 所有木棍都已經被用于拼裝原始木棍,
                            // 即嘗試成功。
                            if ( i >= cutnum ) {
                                    //printf("i >= cutnum, t = %d\n", t);
                                    return 1;
                            }
                            used[ i ] = 1;
                            //choice[ t ] = i;
                            // 任一原始木棍中,必含有剩余木棍中最長的那根
                            if ( dfs(orglen-cutlen[i], i+1, t+1) ) {
                                    return 1;
                            }
                            used[ i ] = 0;
                            return 0;
                    }
                    for ( i = b; i < cutnum; ++i ) {
                            if (    (used[ i ]) ||          // 木棍沒用過
                                    (c < cutlen[i]) ||      // 剩余長度能容納此木棍
                                    (prelen == cutlen[i])   // 此長度的木棍已經試過,不行,無需再試
                            ) {
                                    continue;
                            }
                            used[ i ] = 1;
                            //choice[ t ] = i;

                            // 下次嘗試從后面的木棍開始,前面的木棍無需考慮
                            if ( dfs(c-cutlen[i], i+1, t+1) ) {
                                    return 1;
                            }

                            prelen = cutlen[ i ];

                            used[ i ] = 0;

                            // 存在一木棍剛好可以符合剩余長度,則不再嘗試其它
                            if ( c == cutlen[ i ] ) {
                                    return 0;
                            }
                    }
                    return 0;
            }

            int find(){
                    // 原始木棍長度必為總長度的約數
                    if ( 0 != totlen % orglen ) {
                            return 0;
                    }
                    orgnum = totlen / orglen;
                    memset(used, 0, sizeof(used));
                    return dfs(0, 0, 0);
            }

            int main() {
                    int i;
                    while ( (1 == scanf("%d", &cutnum)) && (0 < cutnum) ) {
                            totlen = 0;
                            for ( i = 0; i < cutnum; ++i ) {
                                    scanf("%d", cutlen+i);
                                    totlen += cutlen[ i ];
                            }
                            // 將木棍從長到短排序,然后從長到短搜索之
                            qsort(cutlen, cutnum, sizeof(cutlen[0]), cmp);
                            // 原始木棍長度必大于等于最長木棍長度,
                            // 小于等于木棍總長度,
                            // 從短到長枚舉原始木棍長度。
                            for ( orglen = cutlen[0]; (orglen < totlen) && (! find()); ++orglen ) {
                            }
                            printf("%d\n", orglen);
                            //if ( orglen != totlen ) {
                            //        for ( i = 0; i < cutnum; ++i ) {
                            //                printf("--%d--", choice[ i ]);
                            //        }
                            //        printf("\n");
                            //}
                    }
                    return 0;
            }
            */



            /**********************************************************
            版本一:
            EOJ  1981  TLE
            POJ  1011  16MS  164K
            HDOJ 1455  0MS   188K
            UVA  307   TLE
            */

            /*
            #include <stdio.h>
            #include <stdlib.h>
            #include <string.h>

            #define  L  70

            int cutnum;                 // 木棍的數量
            int cutlen[ L ], totlen;    // 木棍的各自長度,及總長度

            int cmp( const void *pa, const void *pb) {
                    return (*((int*)pb)) - (*((int*)pa));
            }

            int orglen, orgnum; // 原始木棍的長度及其數量
            int used[ L ];      // dfs 時標記某木棍是否已經被用于組裝原始木棍
            //int choice[ L ];

                    // 嘗試拼裝某原始木棍,
                    // 此次嘗試之前,
                    // 已經使用了 t 根木棍,
                    // 已經拼裝出了原始木棍的部分長度,剩余 c 長度需要拼裝,
                    // 此次嘗試,只需要從第 b 根木棍開始。
            int dfs(int c, int b, int t) {
                    int i, prelen = -1;
                    if ( 0 == c ) {
                            // 剩余長度為 0,則開始拼裝下一根原始木棍,
                            // 需擴大嘗試的范圍。
                            for ( i = 0; (i < cutnum) && (used[i]); ++i ) {
                            }
                            // 所有木棍都已經被用于拼裝原始木棍,
                            // 即嘗試成功。
                            if ( i >= cutnum ) {
                                    //printf("i >= cutnum, t = %d\n", t);
                                    return 1;
                            }
                            used[ i ] = 1;
                            //choice[ t ] = i;
                            // 任一原始木棍中,必含有剩余木棍中最長的那根
                            if ( dfs(orglen-cutlen[i], i+1, t+1) ) {
                                    return 1;
                            }
                            used[ i ] = 0;
                            return 0;
                    }
                    for ( i = b; i < cutnum; ++i ) {
                            if (    (used[ i ]) ||          // 木棍沒用過
                                    (c < cutlen[i]) ||      // 剩余長度能容納此木棍
                                    (prelen == cutlen[i])   // 此長度的木棍已經試過,不行,無需再試
                            ) {
                                    continue;
                            }
                            used[ i ] = 1;
                            //choice[ t ] = i;

                            // 下次嘗試從后面的木棍開始,前面的木棍無需考慮
                            if ( dfs(c-cutlen[i], i+1, t+1) ) {
                                    return 1;
                            }
                            prelen = cutlen[ i ];

                            used[ i ] = 0;
                    }
                    return 0;
            }

            int find(){
                    // 原始木棍長度必為總長度的約數
                    if ( 0 != totlen % orglen ) {
                            return 0;
                    }
                    orgnum = totlen / orglen;
                    memset(used, 0, sizeof(used));
                    return dfs(0, 0, 0);
            }

            int main() {
                    int i;
                    while ( (1 == scanf("%d", &cutnum)) && (0 < cutnum) ) {
                            totlen = 0;
                            for ( i = 0; i < cutnum; ++i ) {
                                    scanf("%d", cutlen+i);
                                    totlen += cutlen[ i ];
                            }
                            // 將木棍從長到短排序,然后從長到短搜索之
                            qsort(cutlen, cutnum, sizeof(cutlen[0]), cmp);
                            // 原始木棍長度必大于等于最長木棍長度,
                            // 小于等于木棍總長度,
                            // 從短到長枚舉原始木棍長度。
                            for ( orglen = cutlen[0]; (orglen < totlen) && (! find()); ++orglen ) {
                            }
                            printf("%d\n", orglen);
                            //if ( orglen != totlen ) {
                            //        for ( i = 0; i < cutnum; ++i ) {
                            //                printf("--%d--", choice[ i ]);
                            //        }
                            //        printf("\n");
                            //}
                    }
                    return 0;
            }
            */

            posted on 2012-04-21 10:47 coreBugZJ 閱讀(3186) 評論(0)  編輯 收藏 引用 所屬分類: ACMAlgorithm課內作業

            久久久久久国产精品无码下载 | 久久天天躁狠狠躁夜夜avapp| 久久男人AV资源网站| 久久久网中文字幕| 久久精品国产清自在天天线| 久久婷婷成人综合色综合| 久久综合久久综合久久| 国产精品99久久久精品无码| 久久99国产精品久久99果冻传媒| 国产精品gz久久久| 久久亚洲欧美国产精品 | 一本一本久久A久久综合精品| 国内精品久久久久久久97牛牛| 99久久精品无码一区二区毛片| 国产精品久久久久蜜芽| 久久国产一片免费观看| 国内精品久久久久久99蜜桃| 亚洲国产成人久久综合一区77| 91久久精品91久久性色| 久久精品国产色蜜蜜麻豆| 久久精品国产精品亜洲毛片| 97热久久免费频精品99| 久久亚洲精品无码aⅴ大香| 国产精品久久久久乳精品爆| 久久久久久九九99精品| 久久久久99这里有精品10| 久久99精品久久久久久齐齐 | 一本色道久久88综合日韩精品 | 国产精品一区二区久久| 欧美午夜精品久久久久免费视| 久久综合久久性久99毛片| 国产成人久久久精品二区三区| 好久久免费视频高清| 久久午夜羞羞影院免费观看| 伊人久久大香线蕉亚洲五月天| 深夜久久AAAAA级毛片免费看| 久久精品国产精品亚洲艾草网美妙| 亚洲成人精品久久| 久久综合九色综合久99| 精品午夜久久福利大片| 免费精品99久久国产综合精品|