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.mzph.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;简称…