LeetCode 945. 使數組唯一的最小增量

来源:https://www.cnblogs.com/izhoujie/archive/2020/03/22/12548529.html
-Advertisement-
Play Games

我的LeetCode刷題源碼[GitHub]:https://github.com/izhoujie/Algorithmcii LeetCode 945. 使數組唯一的最小增量 題目 給定整數數組 A,每次 move 操作將會選擇任意 A[i],並將其遞增 1。 返回使 A 中的每個值都是唯一的最少 ...


我的LeetCode刷題源碼[GitHub]:https://github.com/izhoujie/Algorithmcii

LeetCode 945. 使數組唯一的最小增量

題目

給定整數數組 A,每次 move 操作將會選擇任意 A[i],並將其遞增 1。

返回使 A 中的每個值都是唯一的最少操作次數。

示例 1:

輸入:[1,2,2]
輸出:1
解釋:經過一次 move 操作,數組將變為 [1, 2, 3]。
示例 2:

輸入:[3,2,1,2,1,7]
輸出:6
解釋:經過 6 次 move 操作,數組將變為 [3, 4, 1, 2, 5, 7]。

可以看出 5 次或 5 次以下的 move 操作是不能讓數組的每個值唯一的。

提示:

0 <= A.length <= 40000
0 <= A[i] < 40000

來源:力扣(LeetCode)
鏈接:https://leetcode-cn.com/problems/minimum-increment-to-make-array-unique
著作權歸領扣網路所有。商業轉載請聯繫官方授權,非商業轉載請註明出處。

解題思路

思路1-先排序,再從左向右累加每兩個臨近數需要的+1操作數;

  1. Arrays.sort(A)排序;
  2. 順次比較並記錄將後一個數變為前一個數+1數所需要的操作數;

例子:假如排序後是1123455那麼從第二個數開始:

  • 第一次:1223455 此時move+=2-1
  • 第二次:1233455 此時move+=3-2
  • 第三次:1234455 此時move+=4-3
  • 第四次:1234555 無需操作
  • 第五次:1234565 此時move+=6-5
  • 第六次:1234567 此時move+=7-6
    其中:
  • 時間複雜度:O(NlogN) N=A.length
  • 空間複雜度:O(1)

思路2-先統計順次進行操作數的累加,每次需要累加+1操作的次數是相同數的個數;

  1. 創建新數組new int[40001],然後將A中每個數作為下標進行統計;
  2. 遍歷新數組,1中統計到的相同數大於0的,其-1後的數就是這些數需要進行+1操作的數,並把這些+1操作後的數累加給下一個統計數,通過每次-1來使得最終數都不相同;
  3. 遍歷完後需要再檢查一下最大下標數的個數,若大於1,其中-1個數都需要進行+1操作,直接使用1-n的求和公式即可;
  • 時間複雜度:O(N) N=max(A.length,max(A))
  • 空間複雜度:O(40001)即O(1)

Tips:第一步的統計其實隱含了排序,利用了自然數的特性,下標天然有序是數組很容易被忽略的一個特性,比如字母(通過char的 -'a'操作)轉數組去統計就避免了額外排序;

思路3-路徑壓縮;(來自LeetCode評論區,很秀...)

  1. 創建新數組new int[80000](初始化值-1),因為路徑壓縮不同於方法2中的統計(或者說是統計壓縮),這裡壓縮的是+1的操作,但是+1後的數需要新數組去記錄,若A中所有值都是39999,最後的最大數將是79999;
  2. 開始遍歷併進行路徑點位記錄findPath,這是個遞歸方法,可能比較繞,單獨分析下:
private int findPath(int i) {
	// 初次遇到點位,記錄值並返回,此時j=0
	if (path[i] == -1) {
		path[i] = i;
		return i;
	} else {
		// 若i有記錄,則向後找path[i] + 1位置的值,並最終遞歸更新路徑值
		path[i] = findPath(path[i] + 1);
		return path[i];
	}
}

對於例子:A{1,1,2,3,5,5,2},對應的路徑數組初始化的路徑值均為-1

  • 0下標1的路徑值為-1,執行後更新為1並返回1,此時move+=1-1,對應1不需要+1操作;
  • 1下標1的路徑值因為已經被標記為1了,所以往後找1+1的路徑值,此時找到的2的路徑值為-1,更新路徑1和2的值都為2並返回,最後move+=2-1,對應1需要1次+1操作;
  • 2下標2的路徑值為2,往後找2+1的路徑值得到-1,此時將路徑1,2,3的路徑值都更新為3,最後move+=3-2,對應2需要1次+1操作;
  • 3下標3的路徑值為3,往後找3+1的路徑值得到-1,此時將路徑1,2,3,4的路徑值都更新為4,最後move+=4-3,對應3需要1次+1操作;
  • 4下標5的路徑值為-1,執行後更新為5並返回5,此時move+=5-5,對應5不需要+1操作;
  • 5下標5的路徑值為5,往後找5+1的路徑值得到-1,此時將路徑1,2,3,4,5,6的路徑值都更新為6,最後move+=6-5,對應5需要1次+1操作;
  • 6下標2的路徑值為6,往後找6+1的路徑值得到-1,此時將路徑1,2,3,4,5,6,7的路徑值都更新為7,最後move+=7-2,對應2需要5次+1操作;

所謂的路徑壓縮其實是記錄了當前已遍曆數經過+1操作後中的最大數,便於後面根據路徑直接找到最大數併在此基礎上計算需要+1的次數

演算法源碼示例

package leetcode;

import java.util.Arrays;

/**
 * @author ZhouJie
 * @date 2020年3月22日 下午8:04:55 
 * @Description: 945. 使數組唯一的最小增量
 *
 */
public class LeetCode_0945 {

}

class Solution_0945 {
	/**
	 * @author: ZhouJie
	 * @date: 2020年3月22日 下午8:08:06 
	 * @param: @param A
	 * @param: @return
	 * @return: int
	 * @Description: 1-先排序,再從左向右累加每兩個臨近數需要的+1操作數;
	 * 				時間複雜度:O(NlogN) N=A.length
	 * 				空間複雜度:O(1)
	 *
	 */
	public int minIncrementForUnique_1(int[] A) {
		int len = 0, move = 0;
		if (A == null || (len = A.length) < 2) {
			return move;
		}
		Arrays.sort(A);
		for (int i = 1; i < len; i++) {
			// 若當前值小於等於前一個值,說明需要進行+1操作,+1操作的次數就等於差值再+1,此外還需要更新當前值為前一個值+1
			if (A[i] <= A[i - 1]) {
				move += A[i - 1] - A[i] + 1;
				A[i] = A[i - 1] + 1;
			}
		}
		return move;
	}

	/**
	 * @author: ZhouJie
	 * @date: 2020年3月22日 下午8:15:17 
	 * @param: @param A
	 * @param: @return
	 * @return: int
	 * @Description: 2-先統計再由小到大進行操作數的累加,每次需要累加+1次數的是相同數的個數-1;
	 * 				其實統計這一步隱含了排序(自然數的特性),這也是比1方法快的關鍵原因
	 * 				時間複雜度:O(N) N=max(A.length,max(A))
	 * 				空間複雜度:O(40001)即O(1)
	 */
	public int minIncrementForUnique_2(int[] A) {
		int move = 0;
		if (A == null || A.length < 2) {
			return move;
		}
		// 因為最大數是3999,若+1為40000,需要用到40000索引
		int[] statistics = new int[40001];
		// 記錄最大數,用作遍歷statistics的右邊界
		int max = 0;
		for (int i : A) {
			statistics[i]++;
			max = Math.max(max, i);
		}
		max++;
		// max是A中最終可能的最大值
		for (int i = 0; i < max; i++) {
			// 若A中statistics[i]的個數大於1,說明statistics[i]-1個數需要進行+1操作,
			// 這一步只是給statistics[i]-1個數各進行了一次+1操作,後續的+1交給statistics[i+1]去完成(遞歸)
			if (statistics[i] > 1) {
				move += statistics[i] - 1;
				statistics[i + 1] += statistics[i] - 1;
			}
		}
		// 若statistics[max]的個數大於1,則statistics[max]-1個數(記為n個)需要進行+1操作;
		// 這n個數依次需要進行+1的次數為1、2、3、4....n,即對1-n求和,直接使用求和公式
		if (statistics[max] > 1) {
			int n = statistics[max] - 1;
			move += n * (n + 1) / 2;
		}
		return move;
	}

	/**
	 * @author: ZhouJie
	 * @date: 2020年3月22日 下午8:36:13 
	 * @param: @param A
	 * @param: @return
	 * @return: int
	 * @Description: 1-路徑壓縮;(來自LeetCode評論區,很秀...)
	 * 				時間複雜度:O(N) N=A.length
	 * 				空間複雜度:O(80000)即O(1)
	 * 				因為findPath每次可能更改多個點位的值,所以效率沒有方法2高
	 */

	// 若A中所有數都相等且都為39999,則+1操作完成時,最大值將為79999
	int[] path = new int[80000];

	public int minIncrementForUnique_3(int[] A) {
		int move = 0;
		if (A == null || A.length < 2) {
			return move;
		}
		// -1為空地址標記,與A中數不同即可
		Arrays.fill(path, -1);
		for (int i : A) {
			int j = findPath(i);
			move += j - i;
		}
		return move;
	}

	/**
	 * @author: ZhouJie
	 * @date: 2020年3月22日 下午8:49:55 
	 * @param: @param i
	 * @param: @return
	 * @return: int
	 * @Description: 路徑壓縮核心
	 *
	 */
	private int findPath(int i) {
		// 初次遇到點位,記錄值並返回,此時j=0
		if (path[i] == -1) {
			path[i] = i;
			return i;
		} else {
			// 若i有記錄,則向後找path[i] + 1位置的值,並最終遞歸更新路徑值
			path[i] = findPath(path[i] + 1);
			return path[i];
		}
	}
}


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

-Advertisement-
Play Games
更多相關文章
  • 距離Java 8發佈已經過去了7、8年的時間,Java 14也剛剛發佈。Java 8中關於函數式編程和新增的Stream流API至今飽受“爭議”。 如果你不曾使用Stream流,那麼當你見到Stream操作時一定對它發出過鄙夷的聲音,併在心裡說出“這都寫的什麼玩意兒”。 如果你熱衷於使用Stream ...
  • 恢復內容開始 1.背景:現在很多app或者網站都想要接入微信登錄,可以使用戶不需要註冊就能快速使用APP或網站。 2.微信登錄需要一些前置操作 2.1 搜索:微信開放平臺 鏈接:https://open.weixin.qq.com/ 2.2 註冊成功,獲取到開發所需要的appID和appsecret ...
  • elastic4s是elasticsearch一個第三方開發的scala語言終端工具庫(Elastic4s is a concise, idiomatic, reactive, type safe Scala client for Elasticsearch.)。scala用戶可以用elastic4 ...
  • 1. 不可變的PyIntObject "Python源碼剖析 對象初探" 我們對 PyIntObject 已經有了初步的瞭解。 Python 中的對象可以分為固定長度和可變長度兩種類型。除此之外,也可以按照可變和不可變進行劃分。 PyIntObject 則屬於長度固定且不可變的對象。相比其他的對象而 ...
  • 【qdox】Java 代碼解析利器 QDox 前言 最近在寫 maven 插件,涉及到了 java 代碼解析這塊內容。需要解析 java 源碼,然後對於類中的不同部分進行處理。發現手寫還是很難的,找了一圈發現了兩個不錯的工具可以使用,一個是 "javaparser" ,另一個是 "qdox" 。個人 ...
  • xlrd openpyxl 庫 應該是自帶了,反正我是沒安裝,csv也可製作表格,關於csv以後介紹 基本操作 import xlrd book = xlrd.open_workbook('filename') book.nsheets # 表單數量 book.sheet_names() # 表單名 ...
  • 通過Windows命令行編譯運行C語言源文件 1. 檢查C語言的編譯環境是否配置成功 + win+r 輸入 打開dos視窗,輸入命令 ,查看是否有內容輸出 + 如果有以下內容輸出,則說明配置成功,可以進行下一步 + 如果輸入該命令後,dos視窗輸出以下內容 則說明編譯環境配置失敗,需要重新配置 + ...
  • 1. Mapper映射代理介面 MyBatis基於代理機制,可以讓我們無需在編寫Dao層的介面,我們可以把以前的Dao層的IXxxDao介面直接定義成符合規則的Mapper介面。 具體實現步驟如下: ① 在項目mapper文件目錄下創建對應的Mapper介面,介面命令必須是以Mapper結尾,名字是 ...
一周排行
    -Advertisement-
    Play Games
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...
  • 目錄前言PostgreSql安裝測試額外Nuget安裝Person.cs模擬運行Navicate連postgresql解決方案Garnet為什麼要選擇Garnet而不是RedisRedis不再開源Windows版的Redis是由微軟維護的Windows Redis版本老舊,後續可能不再更新Garne ...
  • C#TMS系統代碼-聯表報表學習 領導被裁了之後很快就有人上任了,幾乎是無縫銜接,很難讓我不想到這早就決定好了。我的職責沒有任何變化。感受下來這個系統封裝程度很高,我只要會調用方法就行。這個系統交付之後不會有太多問題,更多應該是做小需求,有大的開發任務應該也是第二期的事,嗯?怎麼感覺我變成運維了?而 ...
  • 我在隨筆《EAV模型(實體-屬性-值)的設計和低代碼的處理方案(1)》中介紹了一些基本的EAV模型設計知識和基於Winform場景下低代碼(或者說無代碼)的一些實現思路,在本篇隨筆中,我們來分析一下這種針對通用業務,且只需定義就能構建業務模塊存儲和界面的解決方案,其中的數據查詢處理的操作。 ...
  • 對某個遠程伺服器啟用和設置NTP服務(Windows系統) 打開註冊表 HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\W32Time\TimeProviders\NtpServer 將 Enabled 的值設置為 1,這將啟用NTP伺服器功 ...
  • title: Django信號與擴展:深入理解與實踐 date: 2024/5/15 22:40:52 updated: 2024/5/15 22:40:52 categories: 後端開發 tags: Django 信號 松耦合 觀察者 擴展 安全 性能 第一部分:Django信號基礎 Djan ...
  • 使用xadmin2遇到的問題&解決 環境配置: 使用的模塊版本: 關聯的包 Django 3.2.15 mysqlclient 2.2.4 xadmin 2.0.1 django-crispy-forms >= 1.6.0 django-import-export >= 0.5.1 django-r ...
  • 今天我打算整點兒不一樣的內容,通過之前學習的TransformerMap和LazyMap鏈,想搞點不一樣的,所以我關註了另外一條鏈DefaultedMap鏈,主要調用鏈為: 調用鏈詳細描述: ObjectInputStream.readObject() DefaultedMap.readObject ...
  • 後端應用級開發者該如何擁抱 AI GC?就是在這樣的一個大的浪潮下,我們的傳統的應用級開發者。我們該如何選擇職業或者是如何去快速轉型,跟上這樣的一個行業的一個浪潮? 0 AI金字塔模型 越往上它的整個難度就是職業機會也好,或者說是整個的這個運作也好,它的難度會越大,然後越往下機會就會越多,所以這是一 ...
  • @Autowired是Spring框架提供的註解,@Resource是Java EE 5規範提供的註解。 @Autowired預設按照類型自動裝配,而@Resource預設按照名稱自動裝配。 @Autowired支持@Qualifier註解來指定裝配哪一個具有相同類型的bean,而@Resourc... ...