數據結構 集合_集合實例(集合覆蓋)

来源:http://www.cnblogs.com/idreamo/archive/2017/12/03/7953087.html
-Advertisement-
Play Games

設想從一大群選手中挑選人員組建一支隊伍,每名選手都擁有特定的技能組合。目標是組建出一隻最小的隊伍,使得隊伍整體擁有一組特定的技能組合。也就是說,對於隊伍整體所需要的技能,隊伍中至少有一名選手必須擁有這項技能。假定S為隊伍所必須擁有的技能集合,P為所有待選選手的技能集合。從P中挑選出一些技能組合以構成... ...


集合覆蓋是一種優化求解問題,對很多組合數學和資源選擇問題給出了很好的抽象模型。 問題如下:給定一個集合S,集合P由集合S的子集A1到An組成,集合C由集合P中的一個或多個子集組成。如果S中的每個成員都包含在C的至少一個子集中則稱集合C覆蓋集合S。此外,C包含的P的子集越少越好。

設想從一大群選手中挑選人員組建一支隊伍,每名選手都擁有特定的技能組合。目標是組建出一隻最小的隊伍,使得隊伍整體擁有一組特定的技能組合。也就是說,對於隊伍整體所需要的技能,隊伍中至少有一名選手必須擁有這項技能。假定S為隊伍所必須擁有的技能集合,P為所有待選選手的技能集合。從P中挑選出一些技能組合以構成C,C必須覆蓋S中所要求的所有技能。重要一點,我們選擇的選手數量必須儘可能少。

針對集合覆蓋的演算法是一種近似演算法,它並不總是獲得最優解。該演算法的工作原理是:

不斷從P中選出一個集合,使其能夠覆蓋S中最多的成員數量。換句話說,該演算法每次都嘗試儘可能早覆蓋S中更多的成員,因此該演算法採用了貪心法的思路。由於每個集合都是從P中選出的,如果P被移除,則它的成員也將從S中移除。當P中剩餘的成員沒有任何集合能夠覆蓋S中的成員時,此時覆蓋集合C就完成了。

讓我們看看對於12種技能的集合S={a,b,c,d,e,f,g,h,i,j,k,l}的最佳覆蓋集。現在考慮有7名待選選手的集合P={A1,...A7}。P中選手擁有的技能集合為:A1={a,b,c,d},A2={e,f,g,h},A3={j,k,l},A4={a,e},A5={b,f,g},A6={c,d,g,h,k,l},A7={l}。最佳覆蓋集應該是C={A1,A2,A3}。這裡給出的演算法選擇的集合是C={A6,A2,A1,A3}(見圖1)。

集合覆蓋問題的函數實現

我們使用函數cover,該函數在集合P的子集A1~An中挑選出能夠覆蓋集合S的近似最優解。該函數有3個參數:

1、members是待覆蓋的集合S;

2、subsets是集合P中的子集;

3、covering作為返回的覆蓋集C。

該函數將修改所傳入的3個參數,因此在調用該函數時,如果有必要話應該保存一份參數的拷貝。

函數執行過程:開始時,covering通過調用set_init先得到初始化。

我們使用迴圈進行迭代,只要members中還有未覆蓋的成員,且subsets中的子集還沒有挑選完,最外層的迴圈就得繼續迭代。

在這個迴圈中,每次迭代時它都在subsets中找出能夠覆蓋到members的最大交集。

然後它將這個集合加到覆蓋集covering中並把它的成員從members中移除(因為這些成員已經被覆蓋,下一次迭代將判斷剩餘的成員能否被覆蓋)。在迴圈的最後,將所選擇的這個集合從subsets中移除(已經選中的要移除)。如果最外層的迴圈因為members不為空而終止迭代,則表示subsets中的集合不可能滿足完全覆蓋members的要求。同樣,如果在迭代期間subsets中的成員無法與members的成員形成交集,則同樣表示subsets中的成員無法滿足完全覆蓋members的要求。函數cover如果找到了一個完全覆蓋解,該函數返回0,參數covering指向這個完全覆蓋解;如果不可能實現完全覆蓋,則返回1;其他情況返回-1。

cover的複雜度為O(m3),這裡的m代表members集合中的初始成員個數。在最壞的情況下,對於members中的每一員,subsets中都只有唯一一個子集與之對應,此時的複雜度是O(m3)。在這種情況下,subsets有 m個子集,set_intesection以O(m)的複雜度執行,因為當計算和members求交集時,subsets的每個子集都只有唯一一個個成員需要遍歷。因此cover的內層迴圈是O(m2)的,而這個迴圈要執行m次。

示例1:集合覆蓋問題的頭文件

#ifndef COVER_H
#define COVER_H

#include "set.h"

typedef struct KSet_
{
    void *key;
    Set set;
}KSet;

int cover(Set *member, Set *subsets, Set *covering);

#endif 

 示例2:集合覆蓋問題的函數實現

#include <stdlib.h>
#include "cover.h"
#include "list.h"
#include "set.h"

int cover(Set *members, Set *subsets, Set *covering)
{
    Set intersection;
    KSet *subset;
    ListElmt *member,*max_member;
    void *data;
    int max_size;
    
    /*初始化覆蓋集covering*/
    set_init(covering,subsets->match,NULL);
    
    while(set_size(members)>0 && set_size(subsets)>0)
    {
        /*找到能夠覆蓋最多members成員的子集*/
        max_size = 0;
        for(member = list_head(subsets);member!=NULL;member=list_next(member))
        {
            if(set_intersection(&intersection, &((KSet *)list_data(member))->set, members) != 0)
                return -1;
            
            if(set_size(&intersection)>max_size)
            {
                max_member = member;
                max_size = set_size(&intersection);
            }
            
            set_destroy(&intersection);
        }
        /*如果不存在交集,那麼就不可能有覆蓋集*/
        if(max_size==0)
            return 1;
        
        /*將被選到的子集插入覆蓋集covering中*/
        subset = (KSet *)list_data(max_member);
        
        if(set_insert(covering,subset) != 0)
            return -1;
        
        /*從members中移除已經被覆蓋的元素*/
        for(member=list_head(&((Kset *)list_data(max_member))->set);member != NULL;
            list_next(member))
        {
            data = list_data(member);
            if(set_remove(members,(void **)data)== 0 && members->destroy != NULL)
                members->destroy(data);
        }
        
        /*從子集集合中刪除已經被選用的子集*/
        if(set_remove(subsets,(void **)&subset) != 0)
            return -1;
    }
    
    /*如果members中仍然存在未被覆蓋的元素,那麼也不可能實現完全覆蓋*/
    if(set_size(members)>0)
        return -1;
    
    return 0;
}

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 本人多年來一直在獨自設計並開發一種“面向表達”的編程語言——S#,以求達到數據即程式、程式即數據的最高境界,可以包容大多數慣用的語言特性。直至今天初步成形,特此在博客園上開篇介紹,通過分享和交流進一步發揚光大。 ...
  • 一 抽象類 描述一個事物,由於沒有足夠的信息,這時就將這個事物稱為抽象事物。abstract為抽象關鍵字,被其聲明的類稱為抽象類,其聲明的方法稱為抽象方法。 抽象屬性聲明不提供屬性訪問器的實現,它只聲明該類支持的屬性,而訪問器的實現留給派生類。 抽象方法聲明不提供方法的實現,他必須是一個空方法,而將 ...
  • 最近需要向客戶發送一些宣傳資料,Excel列表裡面有一兩百個記錄,本來想手寫就算了,估摸著也花不了多少時間,不過寫完一個信封我就後悔了,整天敲著鍵盤,書寫的字太難看了,而且感覺手還是有點累。才第一個啊,想著後面還有那麼多,感覺整個人頭都大了,只好放棄,太沒技術含量了。然後尋找有無一些套打的的軟體,不... ...
  • jquery.qqFace.js使用方法 引用 <script src="~/Content/qqFace/js/jquery.qqFace.js?v=3"></script> <script src="~/Content/qqFace/js/jquery-browser.js"></script> ...
  • 上篇文章介紹了ASP.NET中身份驗證的機制與流程,本文將使用代碼的來介紹如何實現第三方賬戶驗證與雙因數驗證。 本章主要內容有: ● 實現基於微軟賬戶的第三方身份驗證 ● 實現雙因數身份驗證 ● 驗證碼機制 實現基於微軟賬戶的第三方身份驗證 在微軟提供的ASP.NET MVC模板代碼中,預設添加了微 ...
  • TreeView控制項顯示的內容比較單一,如果需要呈現更多的詳細信息TreeListView是一個不錯的選擇。因此你可以將一個XMl文檔完整的呈現到該控制項中去! ...
  • 總結:資料庫中某張表有問題,導致查詢速度奇慢! 問題排查過程: 1. 視圖問題 ? 2、sql語句問題? 3、沒有建立必須的索引? 後來: 整個sql語句分段調試,發現加入某張表後就變的非常卡! 將該表信息導入新表,刪除該表,使用新表就不卡了! 數據表竟然有問題!!! 頭一次遇到這種稀奇問題,我這個 ...
  • #log4j.rootLogger=Debug,consolelog4j.rootLogger=info,console#控制台輸出 log4j.appender.console=org.apache.log4j.ConsoleAppenderlog4j.appender.console.layou ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...