溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Java中使用貪心算法的示例分析

發(fā)布時間:2021-05-28 11:38:23 來源:億速云 閱讀:105 作者:小新 欄目:開發(fā)技術(shù)

小編給大家分享一下Java中使用貪心算法的示例分析,希望大家閱讀完這篇文章之后都有所收獲,下面讓我們一起去探討吧!

貪心算法

由于貪心算法本身的特殊性,我們在使用貪心算法之前必須要進(jìn)行證明,保證算法滿足貪心選擇性質(zhì)。具體的證明方法無外乎就是通過數(shù)學(xué)歸納法來進(jìn)行證明。但大部分人可能并不喜歡枯燥的公式,因而我這里提供一個使用貪心算法的小技巧。由于貪心算法某種程度上算是動態(tài)規(guī)劃算法的特例,使用條件比較苛刻,因而能夠用動態(tài)規(guī)劃解決的問題盡量都是用動態(tài)規(guī)劃來進(jìn)行先解決,如果在用完動態(tài)規(guī)劃之后,提交時發(fā)現(xiàn)問題超時,并且進(jìn)行狀態(tài)壓縮之后仍然超時,此時我們就可以**考慮使用貪心算法來進(jìn)行解決。**最后強調(diào)一下,我們在使用貪心算法之前,如果要保證解法的絕對正確,一定要對問題進(jìn)行證明,切記,切記??!

下邊我們以區(qū)間調(diào)度問題為例,來講一下貪心算法到底該如何取用。

區(qū)間調(diào)度問題

問題描述:

給你很多形如 [start, end] 的閉區(qū)間,請你設(shè)計一個算法,算出這些區(qū)間中最多有幾個互不相交的區(qū)間。

舉個例子,intvs = [[1,3], [2,4], [3,6]],這些區(qū)間最多有 2 個區(qū)間互不相交,即 [[1,3], [3,6]],你的算法應(yīng)該返回 2。注意邊界相同并不算相交。

這個問題大眼一看好像有很多貪心策略可供選擇,比如我們可以選擇區(qū)間最短的?或者選擇開始最早的?。。。

但是上面幾種策略,我們都可以比較容易的舉出反例來排除,同時這也是貪心算法的另一個小技巧--雖然好多時候直接證明貪心策略的正確性很難,但是我們可以從反證法入手,對貪心策略進(jìn)行證偽,排除許多錯誤的貪心策略。??

好了,說了這么多,那針對該問題正確的貪心策略到底是哪個?

其實正確的思路也比較簡單,可以分成下面三步:

  1. 從區(qū)間集合中選擇一個區(qū)間 x,這個 x 是所有區(qū)間中結(jié)束最早的(end 最小)。

  2. 把所有與 x 區(qū)間相交的區(qū)間從區(qū)間集合中刪除掉。

  3. 重復(fù) 1 和 2,直到區(qū)間集合為空。之前選出的那些 x 的集合就是最大的不想交子集。

這個思路實現(xiàn)成算法的話,可以按照每個區(qū)間的 end 數(shù)值進(jìn)行升序排序,因為這樣處理以后實現(xiàn)步驟 1 和步驟 2 就會容易很多。

我們通過下面這個動圖來輔助理解其整個過程。

Java中使用貪心算法的示例分析

由于我們在計數(shù)之前進(jìn)行了排序,所以所有與 x 相交的區(qū)間必然會和 x 的 end 相交;如果一個區(qū)間不想與 x 的 end 相交,它的 start 必須要大于或者等于 x 的 end。

具體實現(xiàn)的代碼如下:

 public int eraseOverlapIntervals(int[][] intervals) {
    if (intervals.length == 0) {
      return 0;
    }
    Arrays.sort(
        intervals,
        new Comparator<int[]>() {
          @Override
          public int compare(int[] o1, int[] o2) {
            return o1[1] - o2[1];
          }
        });
	//排序后的第一個必然可用
    int count = 1;
    int x_end = intervals[0][1];
    for (int[] interval : intervals) {
      if (interval[0] >= x_end) {
        count++;
        x_end = interval[1];
      }
    }
    return count;
  }

應(yīng)用

如果學(xué)會了上面的區(qū)間調(diào)度問題的話,leetCode 上邊有兩個題目,我們便都可以拿下了。

Java中使用貪心算法的示例分析

這個問題大眼一看好像和我們之前講的那個區(qū)間調(diào)度問題毫不相關(guān),但仔細(xì)分析一下,好像是一模一樣的問題,如果最多有 n 個不重疊的區(qū)間,那么就至少需要 n 個箭頭穿透所有區(qū)間。

 Java中使用貪心算法的示例分析

因而問題也就轉(zhuǎn)化成了,尋找不重疊區(qū)間的個數(shù),但我們要注意的一點是,在 intervalSchedule 算法中,如果兩個區(qū)間的邊界觸碰,不算重疊;而按照這道題目的描述,箭頭如果碰到氣球的邊界氣球也會爆炸,所以說相當(dāng)于區(qū)間的邊界觸碰也算重疊。

 Java中使用貪心算法的示例分析

代碼實現(xiàn)如下:

public int findMinArrowShots(int[][] points) {
    if (points.length <= 0) {
      return 0;
    }
    // 在排序的過程中要考慮溢出情況的發(fā)生
    Arrays.sort(points, (a, b) -> Integer.compare(a[1], b[1]));
    int count = 1;
    int x_end = points[0][1];
    for (int[] point : points) {
      if (point[0] > x_end) {
        count++;
        x_end = point[1];
      }
    }
    return count;
  }

總結(jié)

本文主要結(jié)合一個例子,講了貪心算法的使用方式。

貪心算法實現(xiàn)起來容易,但難在證明。因而文中提供了兩個小竅門輔助判斷是否使用貪心算法:

  1. 在使用考慮貪心算法之前,先考慮使用動態(tài)規(guī)劃(考慮狀態(tài)壓縮)解決該問題,如果問題依然超時,則考慮使用貪心算法。

  2. 在確定貪心策略之前,先用一些特殊的例子驗證貪心策略的正確性。對于正確的貪心策略,為了保證算法的絕對正確,要通過數(shù)學(xué)歸納法進(jìn)行驗證。

看完了這篇文章,相信你對“Java中使用貪心算法的示例分析”有了一定的了解,如果想了解更多相關(guān)知識,歡迎關(guān)注億速云行業(yè)資訊頻道,感謝各位的閱讀!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI