一.貪心算法算法
對於一些最優解問題,每一步都作當前的最優選擇,最後獲得的選擇結果就是最終問題的最優解,這樣的問題就適用貪心算法。貪心算法在每一步作出局部的最優選擇,最後獲得整個問題的最優解。顯然,實際問題中存在大量問題並非每一步最優就能最終最優的,如01揹包問題,所以貪心算法解決問題簡化了解決方案,可是獲得的最終結果的可信度不如動態規劃算法或者分治算法高,每每考慮不夠全面。問題可否使用貪心算法解決要根據問題實際分析。數組
二.活動選擇問題spa
有n個須要在同一天使用同一個教室的活動a1,a2,…,an,教室同一時刻只能由一個活動使用。每一個活動ai都有一個開始時間si和結束時間fi 。一旦被選擇後,活動ai就佔據半開時間區間[si,fi)。若是[si,fi]和[sj,fj]互不重疊,ai和aj兩個活動就能夠被安排在這一天。該問題就是要安排這些活動使得儘可能多的活動能不衝突的舉行(最大兼容活動子集)。例以下圖所示的活動集合S,其中各項活動按照結束時間單調遞增排序。code
{a3,a9,a11}是一個兼容的活動子集,但它不是最大子集,由於子集{a1,a4,a8,a11}更大,實際上它是咱們這個問題的最大兼容子集,但它不是惟一的一個{a2,a4,a9,a11}blog
1.動態規劃算法排序
static void Main(string[] args) { //添加兩個活動,一個從0點到0點,一個從24點到24點 //記錄活動start時間的數組 int[] s = { 0, 1, 3, 0, 5, 3, 5, 6, 8, 8, 2, 12, 24 }; //記錄活動end時間的數組 int[] e = { 0, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 24 }; //記錄全部方案的二維數組 //數組的行數和列數對應了活動方案的編號,如result[3,7]表明了第3個到第7個活動的活動最大兼容子集 List<int>[,] result = new List<int>[13, 13]; //爲二維數組賦初值,集合中的數字表明哪些活動組合成最大兼容子集 for (int m = 0; m < 13; m++) { for (int n = 0; n < 13; n++) { result[m, n] = new List<int>(); } } //雙重循環 //依次遍歷存儲全部活動的最大兼容子集,j是最後一個活動編號,i是第一個活動編號 for (int j = 0; j < 13; j++) { for (int i = 0; i < j - 1; i++) { //int集合sij用於存儲計算出的第i到第j個活動的最大兼容子集 List<int> sij = new List<int>(); //循環遍歷每個活動,number是當前指向的活動編號 for (int number = 1; number < s.Length - 1; number++) { //若是當前遍歷到的活動的時間在最後一個活動j的開始時間和第一個活動i的結束時間之間,這個活動就能插入兩個活動之間進行 if (s[number] >= e[i] && e[number] <= s[j]) { sij.Add(number); } } //若是有活動插入到了兩個活動之間 if (sij.Count > 0) { //保存最大兼容子集的活動的數量 int maxCount = 0; //保存最大兼容子集 List<int> tempList = new List<int>(); //循環遍歷插入的活動 foreach (int number in sij) { //計算最大兼容子集的活動的數量 int count = result[i, number].Count + result[number, j].Count + 1; //更新最大兼容子集的活動數量 //可能有多個活動能夠插入兩個活動之間,所以須要判斷哪個活動插入後是從i活動到j活動的最大兼容子集 //將最大兼容子集的活動編號保存起來 if (maxCount < count) { maxCount = count; tempList = result[i, number].Union<int>(result[number, j]).ToList<int>(); tempList.Add(number); } } //更新計算出的i活動到j活動的最大兼容子集 result[i, j] = tempList; } } } //結果取出第0個活動到第12個活動的最大兼容子集便可 List<int> a = result[0, 12]; foreach (int item in a) { Console.Write(item + " "); } Console.ReadKey(); }
2.貪心算法(遞歸解決)遞歸
能夠看到在ActivitySelection方法的for循環中,找到了一個局部最優結果就break跳出循環繼續尋找下一個結果了(遞歸尋找下一個局部最優解),當前這個結果只是當前的最優解,是不是最終問題的最優解並無管,所以貪心算法的結果並非那麼可信string
static void Main(string[] args) { List<int> list = ActivitySelection(1, 11, 0, 24); foreach(int t in list) { Console.Write(t + " "); } Console.ReadKey(); } static int[] s = { 0, 1, 3, 0, 5, 3, 5, 6, 8, 8, 2, 12, 24 }; static int[] e = { 0, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 24 }; /// <summary> /// 計算給定的活動編號之間的活動可否在給定時間內進行 /// </summary> /// <param name="startActivityNumber">開始的活動編號</param> /// <param name="endActivityNumber">結束的活動編號</param> /// <param name="startTime">開始時間</param> /// <param name="endTime">結束時間</param> /// <returns></returns> public static List<int> ActivitySelection(int startActivityNumber,int endActivityNumber,int startTime,int endTime) { //若是活動已經找完或者時間已經找完,結束遞歸調用 if (startActivityNumber > endActivityNumber || startTime >= endTime) return new List<int>(); //記錄能夠插入的活動編號 int tempNumber = 0; //循環遍歷活動編號,看可否在給定時間內進行 for (int number = startActivityNumber; number <= endActivityNumber; number++) { //判斷可否在給定時間內進行 if(s[number] >= startTime && e[number] <= endTime) { tempNumber = number; break; } } //找到了一個能插入的活動後,繼續遞歸找下一個能插入的活動 List<int> list = ActivitySelection(tempNumber + 1, endActivityNumber, e[tempNumber], endTime); //將找到的活動編號添加入集合中 list.Add(tempNumber); //返回找到的結果 return list; }
3.貪心算法(迭代解決)it
迭代的解決方案只須要遍歷全部的活動,而後看活動可否在指定時間內進行,若是能夠的話就將該活動添加進集合,而後將下一個活動能夠開始的時間更新爲剛纔活動的結束時間。io
static void Main(string[] args) { List<int> list = ActivitySelection(1, 11, 0, 24); foreach(int t in list) { Console.Write(t + " "); } Console.ReadKey(); } static int[] s = { 0, 1, 3, 0, 5, 3, 5, 6, 8, 8, 2, 12, 24 }; static int[] e = { 0, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 24 }; /// <summary> /// 計算給定的活動編號之間的活動可否在給定時間內進行 /// </summary> /// <param name="startActivityNumber">開始的活動編號</param> /// <param name="endActivityNumber">結束的活動編號</param> /// <param name="startTime">開始時間</param> /// <param name="endTime">結束時間</param> /// <returns></returns> public static List<int> ActivitySelection(int startActivityNumber,int endActivityNumber,int startTime,int endTime) { //保存結果的list List<int> list = new List<int>(); //循環遍歷活動編號,看可否在給定時間內進行 for (int number = 1; number <= 11; number++) { //判斷可否在給定時間內進行,找到能夠在給定時間內進行的活動後將活動編號加入結果的集合,而且下一個活動的開始時間須要更新 if(s[number] >= startTime && e[number] <= endTime) { list.Add(number); startTime = e[number]; } } //返回找到的結果 return list; }