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

            CSU OJ - 1219: 建食堂 (所有結(jié)點(diǎn)間的最短路徑)

            鏈接:http://acm.csu.edu.cn/OnlineJudge/problem.php?id=1219

            這個(gè)題
            就是求出所有結(jié)點(diǎn)的距離之后,再找出某個(gè)結(jié)點(diǎn),該結(jié)點(diǎn)離其它結(jié)點(diǎn)的最大距離是所有結(jié)點(diǎn)中是最小的...
            解法1:深搜出所有結(jié)點(diǎn)間的距離,但是會超時(shí),即使深搜的過程使用中記憶化搜索(就是用2維數(shù)組保存已經(jīng)搜出的答案,如果后面的搜索需要用到直接使用即可)...
            解法2:Floyd算法,3重循環(huán)直接找出所有結(jié)點(diǎn)之間的最短距離
            解法3:對每一個(gè)結(jié)點(diǎn)應(yīng)用一次迪杰斯特拉算法,找出所有結(jié)點(diǎn)與其它結(jié)點(diǎn)間的最短距離...

            解法2:
            #include <stdio.h>
            #include <string.h>
            #define MAX  (100 + 10)
            #define INF (1000000 + 10)
            int nN, nM;
            int nDis[MAX][MAX];
            void SearchAll()
            {
                for (int k = 0; k < nN; ++k)
                {
                    for (int i = 0; i < nN; ++i)
                    {   
                        for (int j = 0; j < nN; ++j)
                        {
                            if (nDis[i][k] + nDis[k][j] < nDis[i][j])
                            {
                                nDis[i][j] = nDis[j][i] = nDis[i][k] + nDis[k][j];
                            }
                        }
                    }
                }
            }
            int main()
            {
                while (scanf("%d%d", &nN, &nM) == 2)
                {
                    for (int i = 0; i < nN; ++i)
                    {
                        for (int j = 0; j < nN; ++j)
                        {
                            if (i == j)
                            {
                                nDis[i][j] = 0;
                            }
                            else
                            {
                                nDis[i][j] = INF;
                            }
                        }
                    }
                    while (nM--)
                    {
                        int nX, nY, nK;
                        scanf("%d%d%d", &nX, &nY, &nK);
                        nDis[nX][nY] = nDis[nY][nX] = nK;
                    }
                    SearchAll();
                    bool bOk = false;
                    int nMin = 1 << 30;
                    
                    for (int i = 0; i < nN; ++i)
                    {
                        int nTemp = 0;
                        int j = 0;
                        for ( ; j < nN; ++j)
                        {
                            if (i == j) continue;
                            if (nDis[i][j] == INF)
                            {
                                break;
                            }
                            else
                            {
                                if (nDis[i][j] > nTemp)
                                {
                                    nTemp = nDis[i][j];
                                }
                            }
                        }
                        if (j == nN)
                        {
                            bOk = true;
                            if (nTemp < nMin)
                            {
                                nMin = nTemp;
                            }
                        }
                    }
                    
                    if (bOk)
                    {
                        printf("%d\n", nMin);
                    }
                    else
                    {
                        printf("Can not\n");
                    }
                }
                return 0;
            }

            關(guān)于Floyd算法,可以這樣理解...比如剛開始只取2個(gè)結(jié)點(diǎn)i,j,它們的距離一定是dis(i,j),但是還有其它結(jié)點(diǎn),需要把其它結(jié)點(diǎn)也慢慢加進(jìn)來,所以最外層關(guān)于k的循環(huán)意思就是從0至nN-1,把所有其它結(jié)點(diǎn)加進(jìn)來,比如加入0號結(jié)點(diǎn)后,距離dis(i,0)+dis(0,j)可能會比dis(i,j)小,如果是這樣就更新dis(i,j),然后后面加入1號結(jié)點(diǎn)的時(shí)候,實(shí)際上是在已經(jīng)加入0號結(jié)點(diǎn)的基礎(chǔ)上進(jìn)行的處理了,效果變成dis(i,0,1,j),可能是最小的,而且中間的0,1也可能是不存在的,當(dāng)然是在dis(i,j)原本就是最小的情況下...
            這個(gè)算法可以用下面這個(gè)圖片描述...


            解法3:
            #include <stdio.h>
            #include <string.h>
            #define MAX  (100 + 10)
            #define INF (1000000 + 10)
            int nN, nM;
            int nDis[MAX][MAX];
            void Search(int nSource)
            {
                bool bVisit[MAX];
                memset(bVisit, false, sizeof(bVisit));
                bVisit[nSource] = true;
                for (int i = 0; i < nN - 1; ++i)
                {
                    int nMin = INF;
                    int nMinPos = 0;
                    for (int j = 0; j < nN; ++j)
                    {
                        if (!bVisit[j] && nDis[nSource][j] < nMin)
                        {
                            nMin = nDis[nSource][j];
                            nMinPos = j;
                        }
                    }
                    if (bVisit[nMinPos] == false)
                    {
                        bVisit[nMinPos] = true;
                        for (int j = 0; j < nN; ++j)
                        {
                            if (nDis[nSource][nMinPos] + nDis[nMinPos][j] < nDis[nSource][j])
                            {
                                nDis[nSource][j] = nDis[nSource][nMinPos] + nDis[nMinPos][j];
                            }
                        }
                    }
                }
            }
            void SearchAll()
            {
                for (int k = 0; k < nN; ++k)
                {
                    Search(k);
                }
            }
            int main()
            {
                while (scanf("%d%d", &nN, &nM) == 2)
                {
                    for (int i = 0; i < nN; ++i)
                    {
                        for (int j = 0; j < nN; ++j)
                        {
                            if (i == j)
                            {
                                nDis[i][j] = 0;
                            }
                            else
                            {
                                nDis[i][j] = INF;
                            }
                        }
                    }
                    while (nM--)
                    {
                        int nX, nY, nK;
                        scanf("%d%d%d", &nX, &nY, &nK);
                        nDis[nX][nY] = nDis[nY][nX] = nK;
                    }
                    SearchAll();
                    bool bOk = false;
                    int nMin = 1 << 30;
                    for (int i = 0; i < nN; ++i)
                    {
                        int nTemp = 0;
                        int j = 0;
                        for ( ; j < nN; ++j)
                        {
                            if (i == j) continue;
                            if (nDis[i][j] == INF)
                            {
                                break;
                            }
                            else
                            {
                                if (nDis[i][j] > nTemp)
                                {
                                    nTemp = nDis[i][j];
                                }
                            }
                        }
                        if (j == nN)
                        {
                            bOk = true;
                            if (nTemp < nMin)
                            {
                                nMin = nTemp;
                            }
                        }
                    }
                    if (bOk)
                    {
                        printf("%d\n", nMin);
                    }
                    else
                    {
                        printf("Can not\n");
                    }
                }
                return 0;
            }
            迪杰斯特拉算法的核心思想是維護(hù)一個(gè)源點(diǎn)頂點(diǎn)集合,任何最短路徑一定是從這個(gè)頂點(diǎn)集合發(fā)出的...
            初始化時(shí),這個(gè)集合就是源點(diǎn)...
            我們從該其它結(jié)點(diǎn)中選出一個(gè)結(jié)點(diǎn),該結(jié)點(diǎn)到源點(diǎn)的距離最小...
            顯然,這個(gè)距離就是源點(diǎn)到該結(jié)點(diǎn)的最短距離了,我們已經(jīng)找到了答案的一部分了...然后,我們就把該結(jié)點(diǎn)加入前面所說的頂點(diǎn)集合...
            現(xiàn)在頂點(diǎn)集合更新了,我們必須得更新距離了...由于新加入的結(jié)點(diǎn)可能發(fā)出邊使得原來源點(diǎn)到某些結(jié)點(diǎn)的距離更小,也就是我們的源點(diǎn)變大了,邊也變多了,所以我們的最短距離集合的值也必須變化了...
            該算法一直循環(huán)nN-1次,直至所有的點(diǎn)都加入源點(diǎn)頂點(diǎn)集合...

            posted on 2011-12-04 22:20 yx 閱讀(1232) 評論(0)  編輯 收藏 引用 所屬分類: 解題報(bào)告圖論

            <2012年3月>
            26272829123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            導(dǎo)航

            統(tǒng)計(jì)

            公告

            常用鏈接

            留言簿(3)

            隨筆分類

            隨筆檔案

            me

            好友

            同學(xué)

            網(wǎng)友

            搜索

            最新評論

            閱讀排行榜

            評論排行榜

            久久久精品波多野结衣| 亚洲午夜福利精品久久 | 久久夜色精品国产网站| 精品久久久久久无码人妻热 | 99久久国语露脸精品国产| 久久人人爽人人爽人人片AV不| 色婷婷噜噜久久国产精品12p| 一本大道加勒比久久综合| 国产精品岛国久久久久| 久久精品国产亚洲AV无码麻豆| 久久天天躁狠狠躁夜夜网站| 国产aⅴ激情无码久久| 欧美一区二区久久精品| 伊人久久精品无码av一区 | 午夜不卡久久精品无码免费| 狠狠色婷婷久久一区二区| 久久九九兔免费精品6| 熟妇人妻久久中文字幕| 国产精品99精品久久免费| 久久777国产线看观看精品| 97久久精品国产精品青草| 777久久精品一区二区三区无码| 99久久www免费人成精品| 久久AⅤ人妻少妇嫩草影院| 久久这里只有精品首页| 亚洲va中文字幕无码久久不卡| 国产精品9999久久久久| 精品久久国产一区二区三区香蕉| 热久久国产欧美一区二区精品| 怡红院日本一道日本久久 | 国产综合成人久久大片91| 久久这里只有精品视频99| 亚洲国产婷婷香蕉久久久久久| 久久人人添人人爽添人人片牛牛| 久久久久成人精品无码中文字幕| 久久综合九色综合97_久久久| 国产免费久久久久久无码| 区久久AAA片69亚洲| 国产午夜免费高清久久影院| 手机看片久久高清国产日韩 | 久久99精品久久久久久水蜜桃|