• <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, 評論 - 41, 引用 - 0
            數據加載中……

            從集合中枚舉子集

            在很多算法中需要從一個集合中枚舉可能的子集特別在一些窮舉算法中,需要枚舉每種可能的子集,從而計算出最優解。本文將討論一種把子集映射N進制數字的枚舉方法。

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

            ,子集是否可以重復包含某元素

            ,子集的元素是否有序。

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

            條件一

            條件二


            可以重復包含

            有序

            1

            無序

            2

            不可重復包含

            有序

            3

            無序

            4

            表一

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

            情況1:有{a,a}{ab},{a,c},{ba},{b,b},{b,c}{c,a},{cb},{c,c}9種。

            情況2:有{aa},{b,b},{cc}{a,b},{a,c}{b,a}6種。

            情況3:有{ab},{a,c},{b,a}{bc}{c,a},{cb}6種。

            情況4:有{a,b},{a,c}{ba}3種。

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

            可以重復包含,可以超過

            有序

            1

            無序

            2

            不可重復包含

            有序

            3

            無序

            4

            可以重復包含,不可以超

            有序

            5

            無序

            6

            表二

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

            情況1:有{a,a}{a,b},{a,c}{b,a},{bb},{b,c},{ca},{c,b},{cc}9種。

            情況2:有{a,a},{bb},{c,c},{ab}{a,c},{b,a}6種。

            情況3:有{ab},{a,c},{ba}{b,c},{c,a}{cb}6種。

            情況4:有{a,b},{a,c}{ba}3種。

            情況5:有{a,a},{a,b}{ac}{ba},{bc},{c,a},{c,b}7種。

            情況6:有{aa},{a,b},{ac}{b,a}4種。

            比較例1和例2發現情況1,23,4結果一樣,其實在可以超過的條件下,集合某個元素的個數是不起限制作用,結果也就一樣。所以可合并這兩種情況。從分析中知道,枚舉這樣的集合需要知道兩類信息。N種不同的元素和每種元素個數。N種不同的元素可以映射到0至(N-1N整數上。問題就變成了枚舉N個整數。枚舉出來的數字可以映射到原先的元素。N和表示每種元素個數的數組就是需要的信息。

            // 構造函數
            // 輸入參數:max表示集合元素的個數
            // 輸入參賽:ele_num 表示第i個元素的個數
            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個元素就是取R個數,每個數的取值0至(N-1。這樣每個子集對應一個RN進制的數。于是枚舉數0到NR-1,就枚舉出每種可能的子集,然后判斷子集是否滿足條件。

            // 得到下一個子集合
            // 輸出參數:subset得到的子集合
            // 返回值:1表示成功取得,0表示沒有取得,枚舉完畢
            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:每個枚舉數都滿足要求。

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


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

            // 子集合是否滿足可重復,無序條件
            // 輸出參數: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枚舉數的各位數字不能相同。

            // 子集合是否滿足不重復,有序條件
            // 輸出參數: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:枚舉數高位的數字小于低位的數字。

            // 子集合是否滿足不重復,無序條件
            // 輸出參數: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數字在枚舉數出現的次數不能超過該數字對應元素的個數。

            // 子集元素重復,有序,不能超出集合
            // 輸出參數: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。

            // 子集元素重復,無序,不能超出集合
            // 輸出參數: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;
            }


            其他實現見代碼。

            代碼編譯方式:

            g++ SetIter.cpp -D_SETITER_TEST_ 編譯,運行。可看到例1的結果,

            g++ SetIter.cpp -D_SETITERAGENT_TEST_ 編譯,運行,就可以看到例2的結果。
             

             

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

            亚洲精品高清国产一线久久| 久久久WWW免费人成精品| 精品人妻久久久久久888| 97久久天天综合色天天综合色hd| 欧美精品一区二区精品久久 | 久久精品成人免费观看97| 久久精品青青草原伊人| 99久久免费只有精品国产| av色综合久久天堂av色综合在| 国产精品成人无码久久久久久 | 午夜精品久久久久久99热| 国产精品免费久久久久久久久| 亚洲AV无码成人网站久久精品大| 国产精品美女久久久久av爽| 人妻无码αv中文字幕久久| 一本久久免费视频| 久久黄色视频| 久久久久九九精品影院| 久久中文娱乐网| 久久最新精品国产| 久久久久久无码Av成人影院| 亚洲va久久久噜噜噜久久狠狠| 亚洲欧洲精品成人久久曰影片| 夜夜亚洲天天久久| 青青草原综合久久| 一本一道久久精品综合| 亚洲国产成人久久精品影视| 777米奇久久最新地址| 精品国产VA久久久久久久冰 | 中文字幕日本人妻久久久免费 | 久久精品亚洲一区二区三区浴池 | 精产国品久久一二三产区区别| 久久夜色精品国产| 久久久久久亚洲精品无码| 国产成人久久777777| 精品久久久久一区二区三区| 久久精品国产亚洲5555| 久久亚洲精品国产亚洲老地址| 久久久久这里只有精品 | 久久人人爽人人爽人人片AV不| 久久精品国产久精国产果冻传媒|