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

            輸油管道問題 (POJ - 1723)

               先看算導(dǎo)上輸油管道問題的描述:


               這個(gè)題,雖然說給出了井的x,y坐標(biāo),但是要修建的主管道卻只是一條橫向的,而且其余管道也只是到這條管道的豎向距離。
            那么,就轉(zhuǎn)換為確定一條直線 y = m,使得其它個(gè)點(diǎn)到這條直線的距離最多。也許不需要多的提示,大家的直覺就會(huì)想到應(yīng)該
            選所有y值的中點(diǎn)。但是,這個(gè)的證明卻不是那么的明顯。

            證明如下:
               設(shè)所有的y值系列為y1,y2,...,yn,并且假設(shè)這個(gè)是按遞增排列的。我們要求的是Sum = Σ|yi-m|(1<=i<=n),
               
               1)顯然假如選小于y1或者大于yn的y=m都不會(huì)比選y1或者yn更好。
               2)如果選y1或者yn,那么|y1-m|+|yn-m| = |yn-y1|都是一樣的結(jié)果,甚至選y1和yn之間的任意一個(gè)值。
               3)如此繼續(xù)下去,對(duì)于y2和yn,也有2)所描述的性質(zhì)
               4)繼續(xù)到最后,只需要取最中間一對(duì)點(diǎn)之間的值即可,如果n是奇數(shù),那么就是中間的點(diǎn),如果n是偶數(shù),取任意一個(gè)中間
                    點(diǎn)都可以


               通過上面證明,我們可以選取第y(n/2 + 1)作為修建主管道的地方。當(dāng)然這可能是唯一的最優(yōu)選擇,或者無數(shù)個(gè)最優(yōu)選擇中的一個(gè)。
            那么現(xiàn)在已經(jīng)轉(zhuǎn)換為求中位數(shù)了,求中位數(shù)的辦法最簡(jiǎn)單的是對(duì)序列排序然后取中間的即可。算法導(dǎo)論上有一種平均代價(jià)O(n)的辦法,
            思路類似于快速排序,快排的每一次操作都是劃分?jǐn)?shù)組,前小后大,如果我們也這一次次去劃分?jǐn)?shù)組,剛好軸元素處于我們要求的那個(gè)位置
            上那么就達(dá)到我們的目的了,下面的代碼中Select函數(shù)就是求一個(gè)數(shù)組的中位數(shù)。


               對(duì)于POJ 1723題,很顯然y的選擇是中位數(shù)即可,x的選擇需要轉(zhuǎn)換一下也變成求中位數(shù)了。題目中描述,最后要達(dá)到的效果是每個(gè)士
            兵都占成一橫排,而且彼此相鄰,也就是y相同,但是x系列是k,k+1,k+2,...,k+n-1。那么如何從原來的x0,x1,x2,...,x(n-1)移動(dòng)過去了。
            可以簡(jiǎn)單的考慮下,將最左邊的士兵移動(dòng)到k,次左的移動(dòng)到k+1,...,最右邊的移動(dòng)到k+n-1,所需要的移動(dòng)之和一定是最小的。那么我們
            可以將原來的x0-x(n-1)排序,得到x'0,x'1,...,x'(n-1),要求的Sum = Σ|x'i - (k + i)| = Σ|(x'i - i) -  k|,那么要使Sum最小,只需要
            求序列X'i - i的中位數(shù)即可了。

            代碼如下:
            #include <stdio.h>
            #include <stdlib.h>
            #include <algorithm>
            using std::sort;
            using std::swap;
            #define MAX (10000 + 10)
            int Partion(int* pnA, int nLen)
            {
                int i, j;
                for (i = j = 0; i < nLen - 1; ++i)
                {
                    if (pnA[i] < pnA[nLen - 1])
                    {
                        swap(pnA[i], pnA[j++]);
                    }
                }
                swap(pnA[j], pnA[nLen - 1]);
                return j;
            }
            int Select(int* pnA, int nLen, int nIndex)
            {
                if (nLen > 1)
                {
                    int nP = Partion(pnA, nLen);
                    if (nP + 1 == nIndex)
                    {
                        return pnA[nP];
                    }
                    else if (nP + 1 > nIndex)
                    {
                        return  Select(pnA, nP, nIndex);
                    }
                    else
                    {
                        return Select(pnA + nP + 1, nLen - nP - 1, nIndex - nP - 1);
                    }
                }
                else
                {
                    return pnA[0];
                }
            }
            int main()
            {
                int nX[MAX];
                int nY[MAX];
                int nN;
                int i;
                while (scanf("%d", &nN) == 1)
                {
                    for (i = 0; i < nN; ++i)
                    {
                        scanf("%d%d", &nX[i], &nY[i]);
                    }
                    int nMY = Select(nY, nN, nN / 2 + 1);
                    sort(nX, nX + nN);
                    for (i = 0; i < nN; ++i)
                    {
                        nX[i] = nX[i] - i;
                    }
                    int nMX = Select(nX, nN, nN / 2 + 1);
                    int nSum = 0;
                    for (i = 0; i < nN; ++i)
                    {
                        nSum += abs(nX[i] - nMX);
                        nSum += abs(nY[i] - nMY);
                    }
                    printf("%d\n", nSum);
                }
                
                return 0;
            }

            posted on 2012-03-09 14:27 yx 閱讀(2314) 評(píng)論(0)  編輯 收藏 引用 所屬分類: 順序統(tǒng)計(jì)

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

            導(dǎo)航

            統(tǒng)計(jì)

            公告

            常用鏈接

            留言簿(3)

            隨筆分類

            隨筆檔案

            me

            好友

            同學(xué)

            網(wǎng)友

            搜索

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            久久久久国产精品| 伊色综合久久之综合久久| 亚洲∧v久久久无码精品| 亚洲精品蜜桃久久久久久| 国产精品一区二区久久国产 | 日韩va亚洲va欧美va久久| 久久精品中文字幕第23页| 无码人妻精品一区二区三区久久久| 国产精品久久久久jk制服| 一本综合久久国产二区| 久久久久一区二区三区| 亚洲午夜久久久久久噜噜噜| 国内精品久久久久久久久电影网| 久久精品中文无码资源站| 久久国产成人午夜AV影院| 国内精品久久久久久99| 国产亚洲美女精品久久久2020| 亚洲国产精久久久久久久| 日本欧美久久久久免费播放网| 国产精品免费久久久久久久久| 久久99精品久久久久久久不卡| 久久国产劲爆AV内射—百度| 亚洲а∨天堂久久精品| 久久国产免费直播| 亚洲午夜久久久精品影院 | 97久久天天综合色天天综合色hd| 欧洲国产伦久久久久久久| 久久精品国产亚洲一区二区| 伊人久久大香线蕉av不卡| 国产精品美女久久福利网站| 一本色道久久88综合日韩精品| 久久av高潮av无码av喷吹| 国产伊人久久| 久久有码中文字幕| 欧美成人免费观看久久| 成人综合久久精品色婷婷| 久久精品日日躁夜夜躁欧美| 伊人久久综合无码成人网| 久久亚洲春色中文字幕久久久| 国产午夜福利精品久久2021| 9久久9久久精品|