Leetcode 2158. 每天繪制新區域的數量【Plus題】

1.題目基本信息

1.1.題目描述

有一幅細長的畫,可以用數軸來表示。 給你一個長度為 n 、下標從 0 開始的二維整數數組 paint ,其中 paint[i] = [starti, endi] 表示在第 i 天你需要繪制 starti 和 endi 之間的區域。

多次繪制同一區域會導致不均勻,因此每個區域最多只能繪制 一次 。

返回一個長度為 n 的整數數組 worklog,其中 worklog[i] 是你在第 i 天繪制的 新 區域的數量。

1.2.題目地址

https://leetcode.cn/problems/amount-of-new-area-painted-each-day/description/

2.解題方法

2.1.解題思路

帶懶標記的線段樹 / 區間并查集

2.2.解題步驟

并查集思路:

將連續的區間放到同一個連通分量中,同時維護每個連通分量的根為區間最左側的結點

線段樹思路:

整體是一個求區間和的線段樹,主要區別在于每個結點的非0即1,lazy標記區間中元素是否全為0;然后使用區間和求出區間中被填充的位置個數,間接求出空閑的位置,即需要繪畫的區域長度

3.解題代碼

Python代碼【并查集版本】

class UnionFind():def __init__(self):self.roots = {}self.setCnt = 0self.rootSizes = {}def add(self, x:int) -> None:if x not in self.roots:self.roots[x] = xself.setCnt += 1self.rootSizes[x] = 1def find(self, x:int) -> int:if x not in self.roots:self.add(x)return xroot = self.roots[x]while root != self.roots[root]:root = self.roots[root]# 路徑壓縮while x != root:temp = self.roots[x]self.roots[x] = rootx = tempreturn rootdef union(self, x:int, y:int) -> None:rootx, rooty = self.find(x), self.find(y)if rootx != rooty:if rootx < rooty:self.roots[rooty] = rootxself.rootSizes[rootx] += self.rootSizes[rooty]else:self.roots[rootx] = rootyself.rootSizes[rooty] += self.rootSizes[rootx]self.setCnt -= 1class Solution:def amountPainted(self, paint: List[List[int]]) -> List[int]:# 思路1:并查集(每個數字為一個結點,根結點為連續區間的首部位置,所以每次union時都是將大根連接到小根上)uf = UnionFind()result = []for start, end in paint:thisCnt = 0rootStart, rootEnd = uf.find(start), uf.find(end)while rootEnd != rootStart:uf.union(rootEnd, rootEnd - 1)rootEnd = uf.find(rootEnd)thisCnt += 1result.append(thisCnt)return result

Python代碼【線段樹版本】

# ==> 帶懶標記 求范圍和 的線段樹(數組中的元素非0即1)
class SegmentTreeToSumWithLazy():def __init__(self, nums:list[int]):# 初始化線段樹的存儲數組并進行構建二叉平衡線段樹(這里采用平衡二叉樹,也可以用最大堆進行構建)self.n = len(nums)self.tree = [0] * (self.n * 4)   # 二叉平衡樹的范圍為4*n,如果使用最大堆的自底向上,則范圍2*n即可;維護區間和self.lazy = [0] * (self.n * 4)  # 懶標記數組,lazy[node]如果為1,表示node對應對應區間全部為1,反之不確定為多少self.build(nums, 0, 0, self.n - 1)# 基礎方法:構建樹,在node樹中,將nums[start:end+1]中的區間元素進行插入(node、start、end確定一個線段樹結點)def build(self, nums:list[int], node:int, start:int, end:int) -> None:if start == end:self.tree[node] = nums[start]return mid = (end - start) // 2 + start# 構建左子樹self.build(nums, node * 2 + 1, start, mid)# 構建右子樹self.build(nums, node * 2 + 2, mid + 1, end)# 和形式的線段樹;tree[node]記錄nums[start:end+1]之間元素的和self.tree[node] = self.tree[node * 2 + 1] + self.tree[node * 2 + 2]# 結點懶標記向下推送(基礎函數):將node結點的懶標記推送到左右子結點中,并將自身的懶標記清空def pushDown(self, node:int, start:int, end:int) -> None:# 第一步,特殊判斷。node結點的懶標記值為0,無需向下推送,直接退出if self.lazy[node] == 0:return # 第二步,獲取左右結點的結點編號和范圍,更新左右結點的tree和lazy中的值mid = (end - start) // 2 + startleftChild = node * 2 + 1rightChild = node * 2 + 2self.tree[leftChild] = mid - start + 1self.lazy[leftChild] = 1self.tree[rightChild] = end - midself.lazy[rightChild] = 1# 第三步,清空node對應的lazy值self.lazy[node] = 0# 區間修改(基礎方法):在結點node對應的[start,end]區間中,將其中[left,right]區間部分的原數組值全部增加valuedef rangeUpdate(self, value:int, left:int, right:int, node:int, start:int, end:int) -> None:# 第一步,遞歸退出條件# 1.1.node結點區間[start,end]和目標區間[left,right]不存在交集時,直接退出if left > end or right < start:return # 1.2.node結點區間[start,end]被目標區間[left,right]包含,直接更新lazy和tree數組。tree這種node結點對應值自增;在lazy數組中更新node結點的懶標記,lazy[node]自增valueif left <= start and right >= end:self.tree[node] = end - start + 1self.lazy[node] = 1return# 第二步,node區間和目標區間存在重疊的情況下,遞歸處理# 2.1.由于需要分到兩個子區間中進行遞歸操作,所以node對應的lazy值需要向下推送到子結點中;本質就是將node結點的lazy值分配到左右結點中,更新tree和lazy中對應的值self.pushDown(node, start, end)# 2.2.遞歸范圍更新兩個子樹leftChild, rightChild = node * 2 + 1, node * 2 + 2mid = (end - start) // 2 + startself.rangeUpdate(value, left, right, leftChild, start, mid)self.rangeUpdate(value, left, right, rightChild, mid + 1, end)# 第三步,更新當前結點的tree值(node中增加value的范圍不確定,所以通過子結點來更新)self.tree[node] = self.tree[leftChild] + self.tree[rightChild]# 單點修改:調用區間函數rangeUpdate進行def pointUpdate(self, index:int, value:int, node:int, start:int, end:int) -> None:self.rangeUpdate(value, index, index, node, start, end)# 區間查詢(基礎方法):在結點node對應的[start,end]區間中,求[left,right]區間部分的原數組值的和def rangeSum(self, left:int, right:int, node:int, start:int, end:int) -> int:# 第一步,遞歸退出條件# 1.1.node結點區間和目標區間[left,right]不存在交集時,直接退出if left > end or right < start:return 0# 1.2.node結點區間被目標區間[left,right]包含,直接返回tree數組中node結點的值if left <= start and right >= end:return self.tree[node]# 第二步,node區間和目標區間存在重疊的情況下,遞歸處理# 2.1.由于需要分到兩個子區間中進行遞歸操作,所以node對應的lazy值需要向下推送到子結點中;本質就是將node結點的lazy值分配到左右結點中,更新tree和lazy中對應的值self.pushDown(node, start, end)# 2.2.遞歸獲取兩個子樹的范圍和,相加進行返回mid = (end - start) // 2 + startreturn self.rangeSum(left, right, node * 2 + 1, start, mid) + self.rangeSum(left, right, node * 2 + 2, mid + 1, end)class Solution:def amountPainted(self, paint: List[List[int]]) -> List[int]:# 思路2:帶懶標記的線段樹# 參考:Leetcode 307. 區域和檢索 - 數組可修改maxVal = max([b for a, b in paint])nums = [0] * (maxVal + 1)segTree = SegmentTreeToSumWithLazy(nums)result = []for start, end in paint:left, right = start, end - 1fullCnt = segTree.rangeSum(left, right, 0, 0, maxVal)result.append(right - left + 1 - fullCnt)segTree.rangeUpdate(0, left, right, 0, 0, maxVal)return result

4.執行結果

在這里插入圖片描述

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/pingmian/77472.shtml
繁體地址,請注明出處:http://hk.pswp.cn/pingmian/77472.shtml
英文地址,請注明出處:http://en.pswp.cn/pingmian/77472.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Git Flow

Git Flow深度解析&#xff1a;企業級分支管理實戰指南 前言 在持續交付時代&#xff0c;分支策略決定團隊協作效率。Git Flow作為經典的分支管理模型&#xff0c;被Apache、Spring等知名項目采用。2023年JetBrains開發者調查報告顯示&#xff0c;Git Flow仍是中大型項目最常用…

[Swift]pod install成功后運行項目報錯問題error: Sandbox: bash(84760) deny(1)

操作&#xff1a; platform :ios, 14.0target ZKMKAPP do# Comment the next line if you dont want to use dynamic frameworksuse_frameworks!# Pods for ZKMKAPPpod Moyaend pod install成功后運行報錯 報錯&#xff1a; error: Sandbox: bash(84760) deny(1) file-writ…

[管理與領導-129]:向上管理-組織架構、股權架構、業務架構、流程架構,看每個人在組織中的位置和重要性

目錄 一、股權架構&#xff1a;反映所有權與控制權 二、組織架構&#xff1a;定義角色與匯報關系 三、業務架構&#xff1a;定義業務單元與價值鏈 四、流程架構&#xff1a;規范業務運作與協作 五、綜合分析&#xff1a;個人在組織中的綜合影響力 六、案例&#xff1a;某…

小紅書爬蟲,小紅書api,小紅書數據挖掘

背景&#xff1a; 小紅書&#xff08;Xiaohongshu&#xff09;是一款結合社交、購物和內容分享的移動應用&#xff0c;近年來在中國以及全球范圍內擁有大量的用戶群體。小紅書上的內容包括用戶的消費體驗、生活方式、旅行分享、時尚搭配等。通過這些內容&#xff0c;用戶可以了…

玩轉Docker | 使用Docker部署tududi任務管理工具

玩轉Docker | 使用Docker部署tududi任務管理工具 前言一、tududi介紹Tududi簡介核心功能特點二、系統要求環境要求環境檢查Docker版本檢查檢查操作系統版本三、部署tududi服務下載鏡像創建容器創建容器檢查容器狀態檢查服務端口安全設置四、訪問tududi服務訪問tududi首頁登錄tu…

大屏設計與匯報:政務服務可視化實踐

大屏設計與匯報:政務服務可視化實踐 引言 在政務服務數字化轉型浪潮中,大屏設計成為展現業務能力與數據價值的關鍵手段。本文圍繞政務大屏設計,從設計要點、業務邏輯到匯報技巧展開深入探討,為相關從業者提供全面參考。 一、大屏設計核心要點 (一)多維度考量 設計大…

字節(抖音)golang后端

Golang知道哪些并發模式&#xff0c;你覺得哪個更好&#xff0c;為什么 在使用channel的時候有哪些需要考慮和注意的地方 進程和線程的區別 線程里有哪些字段 TCP和UDP的區別&#xff0c;各自的優劣勢 TCP 更適合需要可靠性、順序和連接管理的場景&#xff0c;如文件傳輸和網頁…

Python語法系列博客 · 第6期[特殊字符] 文件讀寫與文本處理基礎

上一期小練習解答&#xff08;第5期回顧&#xff09; ? 練習1&#xff1a;字符串反轉模塊 string_tools.py # string_tools.py def reverse_string(s):return s[::-1]調用&#xff1a; import string_tools print(string_tools.reverse_string("Hello")) # 輸出…

Unity運行時查看日志插件 (IngameDebugConsole)

Unity運行時查看日志插件 (IngameDebugConsole) 文章目錄 Unity運行時查看日志插件 (IngameDebugConsole)一、介紹二、使用步驟1.導入插件2.開始使用 結束 一、介紹 In-game Debug Console插件可以在打包發布以后&#xff0c;程序運行時方便的看到控制臺信息&#xff0c;在一些…

spark-SQL核心編程課后總結

通用加載與保存方式 加載數據&#xff1a;Spark-SQL的 spark.read.load 是通用加載方法&#xff0c;借助 format 指定數據格式&#xff0c;如 csv 、 jdbc 、 json 等&#xff1b; load 用于指定數據路徑&#xff1b; option 在 jdbc 格式時傳入數據庫連接參數。此外&#xff0…

蔡浩宇的AIGC游戲革命:從《原神》到《Whispers》的技術跨越

目錄 引言&#xff1a;游戲行業的AI革命前夜 一、《Whispers》的技術突破與市場挑戰 1.1 多模態AI技術的集成應用 1.2 與傳統游戲的差異化體驗 1.3 面臨的商業化難題 二、從《原神》到《Whispers》的技術演進 2.1 《原神》成功的時代因素分析 2.2 蔡浩宇的技術路線轉變 …

Spring Boot中定時任務Cron表達式的終極指南

Spring Boot中定時任務Cron表達式的終極指南 一、Cron表達式基礎二、Spring Boot中定時任務的實現三、Cron表達式高級用法四、調試與驗證技巧五、常見問題與解決方案六、最佳實踐總結 定時任務是后端開發中實現周期性業務邏輯的核心技術之一。在Spring Boot生態中&#xff0c;結…

國產SMT貼片機自主技術突破解析

內容概要 隨著電子信息產業對精密制造需求的持續升級&#xff0c;國產SMT貼片機的技術突破已成為裝備自主化進程的關鍵節點。本文聚焦設備研發的三大核心領域&#xff1a;高動態運動控制系統通過線性電機與數字信號處理技術的融合&#xff0c;將重復定位精度提升至5μm級別&am…

uni-app 安卓10以上上傳原圖解決方案

在Android 10及以上版本中&#xff0c;由于系統對文件訪問的限制&#xff0c;使用chooseImage并勾選原圖上傳后&#xff0c;返回的是圖片的外部存儲路徑&#xff0c;如&#xff1a;file:///storage/emulated/0/DCIM/Camera/。這種外部存儲路徑&#xff0c;無法直接轉換成所需要…

迭代器模式:統一不同數據結構的遍歷方式

迭代器模式&#xff1a;統一不同數據結構的遍歷方式 一、模式核心&#xff1a;分離數據遍歷與數據表示 在開發中&#xff0c;我們經常需要遍歷不同的數據結構&#xff0c;如數組、鏈表、樹等。若在客戶端代碼中直接編寫遍歷邏輯&#xff0c;不僅會導致代碼冗余&#xff0c;而…

Oracle 如何停止正在運行的 Job

Oracle 如何停止正在運行的 Job 先了解是dbms_job 還是 dbms_scheduler&#xff0c;再確定操作命令。 一 使用 DBMS_JOB 包停止作業&#xff08;適用于舊版 Job&#xff09; 1.1 查看正在運行的 Job SELECT job, what, this_date, this_sec, failures, broken FROM user_j…

真實波幅策略思路

該策略是一種基于ATR&#xff08;Average True Range&#xff09;指標的交易策略&#xff0c;主要用于期貨市場中的日內交易。策略的核心思想是利用ATR指標來識別市場的波動范圍&#xff0c;并結合均線過濾來確定買入和賣出的時機。 交易邏輯思維 1. 數據準備與初始化 - 集合競…

Web3技術如何提升用戶數據保護

在這個信息爆炸的時代&#xff0c;用戶數據保護已成為全球關注的焦點。Web3 技術&#xff0c;作為下一代互聯網的代表&#xff0c;以其去中心化、安全性和用戶主權等特點&#xff0c;為用戶數據保護提供了新的解決方案。本文將探討 Web3 技術如何提升用戶數據保護。 去中心化存…

銀河麒麟系統 達夢8 安裝 dlask 框架后端環境

適配的一套環境為 dmPython2.5.8 dmSQLAlchemy1.4.39 Flask2.0.3 Flask-Cors3.0.10 Flask-SQLAlchemy2.5.1 SQLAlchemy1.4.54 Werkzeug2.2.2其中 # sqlalchemy-dm1.4.39 通過dmdbms目錄內文件進行源碼安裝 (MindSpore) [ma-user python]$pwd /home/syl/dmdbms/drivers/python…

利用 i2c 快速從 Interface 生成 Class

利用 i2c 快速從 Interface 生成 Class&#xff08;支持 TS & ArkTS&#xff09; 在日常 TypeScript 或 ArkTS 開發中&#xff0c;需要根據 interface 定義手動實現對應的 class&#xff0c;這既重復又容易出錯。分享一個命令行工具 —— interface2class&#xff0c;簡稱…