• <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>
            隨筆 - 51, 文章 - 1, 評(píng)論 - 41, 引用 - 0
            數(shù)據(jù)加載中……

            從集合中枚舉子集

            在很多算法中需要從一個(gè)集合中枚舉可能的子集特別在一些窮舉算法中,需要枚舉每種可能的子集,從而計(jì)算出最優(yōu)解。本文將討論一種把子集映射N進(jìn)制數(shù)字的枚舉方法。

            從集合中枚舉子集有許多種情況。這里集合是指廣義的,它可能包含相同的元素。先討論不含相同元素的集合,枚舉問(wèn)題規(guī)定如下:從N個(gè)元素的集合中,取出R個(gè)的元素子集。根據(jù)子集的不同性質(zhì)可分為:

            子集是否可以重復(fù)包含某元素

            子集的元素是否有序。

            上面兩種情況自由組合可分為4種情形,見(jiàn)下表:

            條件一

            條件二


            可以重復(fù)包含

            有序

            1

            無(wú)序

            2

            不可重復(fù)包含

            有序

            3

            無(wú)序

            4

            表一

            1:按照這4種情況,枚舉集合{abc},其中R=2

            情況1:有{aa}{ab}{ac}{ba}{bb}{bc}{ca}{cb}{cc}9種。

            情況2:有{aa}{bb}{cc}{ab}{ac}{ba}6種。

            情況3:有{ab}{ac}{ba}{bc}{ca}{cb}6種。

            情況4:有{ab}{ac}{ba}3種。

            下面討集合包含相同元素,這里的相同元素視為完全等同,可以替換。這樣集合含有兩個(gè)信息,一是含有N各不同的元素,二是每種元素有多少個(gè)。如果每種元素的個(gè)數(shù)為1,就是上面討論的情況。這里增加了新的討論條件,子集重復(fù)包含某元素的個(gè)數(shù)是否可以超過(guò)集合中該元素的個(gè)數(shù)。上一種情況,重復(fù)包含就意味超過(guò)。而在這里,就要分情況處理。

            可以重復(fù)包含,可以超過(guò)

            有序

            1

            無(wú)序

            2

            不可重復(fù)包含

            有序

            3

            無(wú)序

            4

            可以重復(fù)包含,不可以超過(guò)

            有序

            5

            無(wú)序

            6

            表二

            2:按照這6種情況,枚舉集合{aabc},其中R=2

            情況1:有{aa}{ab}{ac}{ba}{bb}{bc}{ca}{cb}{cc}9種。

            情況2:有{aa}{bb}{cc}{ab}{ac}{ba}6種。

            情況3:有{ab}{ac}{ba}{bc}{ca}{cb}6種。

            情況4:有{ab}{ac}{ba}3種。

            情況5:有{aa}{ab}{ac}{ba}{bc}{ca}{cb}7種。

            情況6:有{aa}{ab}{ac}{ba}4種。

            比較例1和例2發(fā)現(xiàn)情況1234結(jié)果一樣,其實(shí)在可以超過(guò)的條件下,集合某個(gè)元素的個(gè)數(shù)是不起限制作用,結(jié)果也就一樣。所以可合并這兩種情況。從分析中知道,枚舉這樣的集合需要知道兩類(lèi)信息。N種不同的元素和每種元素個(gè)數(shù)。N種不同的元素可以映射到0至(N-1N整數(shù)上。問(wèn)題就變成了枚舉N個(gè)整數(shù)。枚舉出來(lái)的數(shù)字可以映射到原先的元素。N和表示每種元素個(gè)數(shù)的數(shù)組就是需要的信息。

            // 構(gòu)造函數(shù)
            // 輸入?yún)?shù):max表示集合元素的個(gè)數(shù)
            // 輸入?yún)①悾篹le_num 表示第i個(gè)元素的個(gè)數(shù)
            CSetIter::CSetIter(unsigned 
            long max, const std::vector<int>& ele_num) :
                m_ele_num(ele_num)
            {
                assert(max 
            == ele_num.size());
                m_max 
            = max;
            }


            枚舉
            R個(gè)元素就是取R個(gè)數(shù),每個(gè)數(shù)的取值0至(N-1。這樣每個(gè)子集對(duì)應(yīng)一個(gè)RN進(jìn)制的數(shù)。于是枚舉數(shù)0到NR-1就枚舉出每種可能的子集,然后判斷子集是否滿(mǎn)足條件。

            // 得到下一個(gè)子集合
            // 輸出參數(shù):subset得到的子集合
            // 返回值:1表示成功取得,0表示沒(méi)有取得,枚舉完畢
            int CSetIter::GetNextSubset(std::vector<int>& subset)
            {

                assert(subset.size() 
            == m_size);
                
            while (m_iter_num < m_iter_max)
                {
                    
            // 判斷是否符合條件
                    
            if ((this->*m_pfnIsSubsetOk)(m_iter_v))
                    {
                        subset 
            = m_iter_v;
                        IncIterNum();
                        return 
            1;
                    }
                    IncIterNum();
                }
                return 
            0;
            }


            下面分別討論這六種情況如何判斷。

            情況1:每個(gè)枚舉數(shù)都滿(mǎn)足要求。

            // 子集合是否滿(mǎn)足可重復(fù),有序條件
            // 輸出參數(shù):subset得到的子集合
            // 返回值:1表示符合,0表示不符合
            int CSetIter::IsMultOrdered(std::vector<int>& subset)
            {
                return 
            1;
            }


            情況
            2:枚舉數(shù)高位的數(shù)字不大于低位的數(shù)字。

            // 子集合是否滿(mǎn)足可重復(fù),無(wú)序條件
            // 輸出參數(shù):subset得到的子集合
            // 返回值:1表示符合,0表示不符合
            int CSetIter::IsMultDisorder(std::vector<int>& subset)
            {
                
            for (int i=0; i<m_size-1; i++)
                {
                    
            if (subset[i] > subset[i+1])
                        return 
            0;
                }
                return 
            1;
            }


            情況
            3枚舉數(shù)的各位數(shù)字不能相同。

            // 子集合是否滿(mǎn)足不重復(fù),有序條件
            // 輸出參數(shù):subset得到的子集合
            // 返回值:1表示符合,0表示不符合
            int CSetIter::IsSingleOrdered(std::vector<int>& subset)
            {
                
            for (int i=0; i<m_size-1; i++)
                {
                    
            for (int j=i+1; j<m_size; j++)
                    {
                        
            if (subset[i] == subset[j])
                            return 
            0;
                    }
                }
                return 
            1;
            }


            情況4:枚舉數(shù)高位的數(shù)字小于低位的數(shù)字。

            // 子集合是否滿(mǎn)足不重復(fù),無(wú)序條件
            // 輸出參數(shù):subset得到的子集合
            // 返回值:1表示符合,0表示不符
            int CSetIter::IsSingleDisorder(std::vector<int>& subset)
            {
                
            for (int i=0; i<m_size-1; i++)
                {
                    
            if (subset[i] >= subset[i+1])
                        return 
            0;
                }
                return 
            1;
            }


            情況
            5數(shù)字在枚舉數(shù)出現(xiàn)的次數(shù)不能超過(guò)該數(shù)字對(duì)應(yīng)元素的個(gè)數(shù)。

            // 子集元素重復(fù),有序,不能超出集合
            // 輸出參數(shù):subset得到的子集合
            // 返回值:1表示符合,0表示不符
            int CSetIter::IsMultOrderedIn(std::vector<int>& subset)
            {
                std::vector
            <int> tmp(m_ele_num.size(), 0);
                
            for (int i=0; i<m_size; i++)
                {
                    tmp[subset[i]]
            ++;
                    
            if (tmp[subset[i]] > m_ele_num[subset[i]])
                        return 
            0;
                }
                return 
            1;
            }


            情況
            6情況5加上情況2

            // 子集元素重復(fù),無(wú)序,不能超出集合
            // 輸出參數(shù):subset得到的子集合
            // 返回值:1表示符合,0表示不符
            int CSetIter::IsMultDisorderIn(std::vector<int>& subset)
            {
                std::vector
            <int> tmp(m_ele_num.size(), 0);
                
            for (int i=0; i<m_size-1; i++)
                {
                    
            if (subset[i] > subset[i+1])
                        return 
            0;
                }
                
            for (int i=0; i<m_size; i++)
                {
                    tmp[subset[i]]
            ++;
                    
            if (tmp[subset[i]] > m_ele_num[subset[i]])
                        return 
            0;
                }
                return 
            1;
            }


            其他實(shí)現(xiàn)見(jiàn)代碼

            代碼編譯方式:

            g++ SetIter.cpp -D_SETITER_TEST_ 編譯,運(yùn)行。可看到例1的結(jié)果,

            g++ SetIter.cpp -D_SETITERAGENT_TEST_ 編譯,運(yùn)行,就可以看到例2的結(jié)果。
             

             

            posted on 2007-11-03 12:36 lemene 閱讀(2299) 評(píng)論(0)  編輯 收藏 引用


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


            亚洲国产成人久久综合一| 久久久高清免费视频| 国产成年无码久久久久毛片| 国产精品久久99| 精品国产综合区久久久久久| 国产精品久久久久久五月尺| 精品国产一区二区三区久久久狼| 日本久久久久久中文字幕| 久久久黄色大片| 久久久中文字幕| 久久精品国产日本波多野结衣| 久久精品国产精品青草| 日日狠狠久久偷偷色综合0| AV色综合久久天堂AV色综合在| 大美女久久久久久j久久| 99精品久久精品一区二区| 99久久伊人精品综合观看| 亚洲乱码精品久久久久..| 久久精品国产一区二区| 国内精品久久久久久久97牛牛 | 久久久精品无码专区不卡| 久久久久亚洲AV片无码下载蜜桃| 久久93精品国产91久久综合| 久久96国产精品久久久| 久久精品国产99国产精品导航 | 青青青伊人色综合久久| 性欧美丰满熟妇XXXX性久久久| 久久久久久国产a免费观看不卡| 国内精品伊人久久久久av一坑 | 国产精品美女久久福利网站| 岛国搬运www久久| 国产精品青草久久久久婷婷| 人妻精品久久无码专区精东影业 | 国产精品久久久香蕉| 久久人人爽人人爽人人片AV麻豆| 爱做久久久久久| 精品久久久久久无码人妻热| 精品国产婷婷久久久| 久久99久久成人免费播放| 久久久久亚洲精品中文字幕 | 久久婷婷色综合一区二区|