LeetCode 443 压缩字符串

字符数组压缩算法详解:实现与分析

一、引言

在处理字符数组时,我们常常遇到需要对连续重复字符进行压缩的场景。这不仅可以节省存储空间,还能提升数据传输效率。本文将深入解析一个经典的字符数组压缩算法,通过详细的实现步骤和原理分析,帮助读者全面理解这一算法。

二、问题详细描述

给定一个字符数组 chars,请使用以下算法进行压缩:

  1. 从一个空字符串 s 开始。

  2. 对于 chars 中的每组连续重复字符:

    • 如果这一组长度为 1,则将字符追加到 s 中。

    • 否则,需要向 s 追加字符,后跟这一组的长度。例如,字符 'a' 连续出现 3 次,则压缩为 "a3"

  3. 压缩后得到的字符串 s 不应该直接返回,需要转储到字符数组 chars 中。如果组长度为 10 或 10 以上,则在 chars 数组中会被拆分为多个字符。例如,长度为 12 的组将被写为 '1''2' 两个字符。

  4. 修改完输入数组后,返回该数组的新长度。

三、解题思路:双指针法

3.1 算法原理

为了高效地在原地完成字符数组的压缩,我们采用双指针技术。双指针法通过两个指针 readwrite 分别负责读取和写入操作,能够在一次遍历中完成压缩,无需额外的空间。

3.2 详细步骤

  1. 初始化指针readwrite 指针都从 0 开始。

  2. 读取字符read 指针用于遍历字符数组,找到每组连续重复字符。

  3. 计数:对于每个字符,计算其连续出现的次数。这通过一个循环实现,每次检查下一个字符是否与当前字符相同,如果是,则计数加 1。

  4. 写入字符:将当前字符写入 write 指针位置。然后根据计数决定是否需要写入数字。

  5. 写入计数:如果计数大于 1,则将计数转换为字符串,并逐个字符写入 write 指针后续位置。例如,计数为 12 时,写入 '1''2'

  6. 移动指针read 指针移动到下一组字符的起始位置,write 指针根据写入的字符数量移动。

3.3 优势

双指针法的优势在于:

  • 高效性:仅需一次遍历即可完成压缩,时间复杂度为 O(n),其中 n 是数组长度。

  • 空间效率:所有操作在原数组上进行,无需额外存储结构,空间复杂度为 O(1)。

四、算法实现

以下是 Java 实现代码:

class Solution {public int compress(char[] chars) {if (chars == null || chars.length == 0) return 0;int write = 0;int n = chars.length;for (int read = 0; read < n; ) {char current = chars[read];int count = 1;// 计算连续字符的长度while (read + count < n && chars[read + count] == current) {count++;}// 写入当前字符chars[write++] = current;// 如果计数大于 1,写入计数if (count > 1) {String s = String.valueOf(count);for (char c : s.toCharArray()) {chars[write++] = c;}}// 移动读取指针到下一组字符read += count;}return write;}
}

五、代码详细解析

5.1 初始化

if (chars == null || chars.length == 0) return 0;
int write = 0;
int n = chars.length;
  • 首先检查输入数组是否为空或长度为 0。如果是,则直接返回 0。

  • 初始化 write 指针为 0,用于跟踪写入位置。

  • 获取数组长度 n,用于后续循环控制。

5.2 外层循环:遍历字符数组

for (int read = 0; read < n; ) {char current = chars[read];int count = 1;
  • read 指针从 0 开始,遍历整个数组。

  • 对于每个 read 位置的字符,将其存储在 current 中,并初始化计数为 1。

5.3 内层循环:计算连续字符长度

while (read + count < n && chars[read + count] == current) {count++;
}
  • 检查 read + count 是否在数组范围内,并且下一个字符是否与 current 相同。

  • 如果是,则计数加 1,直到遇到不同的字符或数组末尾。

5.4 写入字符

chars[write++] = current;
  • 将当前字符写入 write 指针位置,并将 write 指针向前移动一位。

5.5 写入计数

if (count > 1) {String s = String.valueOf(count);for (char c : s.toCharArray()) {chars[write++] = c;}
}
  • 如果计数大于 1,则将计数转换为字符串。

  • 遍历字符串的每个字符,并将其逐个写入数组。例如,计数为 12 时,写入 '1''2'

5.6 移动读取指针

read += count;
  • read 指针移动到下一组字符的起始位置,继续处理下一组字符。

六、算法分析

6.1 时间复杂度

该算法的时间复杂度为 O(n),其中 n 是字符数组的长度。每个字符仅被处理一次,无论是读取还是写入操作。内层循环虽然看似增加了复杂度,但实际上每个字符只被检查一次,因此总的时间复杂度仍然是线性的。

6.2 空间复杂度

空间复杂度为 O(1),因为我们没有使用任何与输入规模相关的额外存储结构。所有操作都在原数组上进行,仅使用了几个变量来辅助操作。

七、示例分析

示例 1

输入chars = ['a', 'a', 'b', 'b', 'c', 'c', 'c']

步骤解析

  1. read = 0current = 'a',计数为 2。

    • 写入 'a'write = 1

    • 写入 '2'write = 2

    • read 移动到 2。

  2. read = 2current = 'b',计数为 2。

    • 写入 'b'write = 3

    • 写入 '2'write = 4

    • read 移动到 4。

  3. read = 4current = 'c',计数为 3。

    • 写入 'c'write = 5

    • 写入 '3'write = 6

    • read 移动到 7,循环结束。

输出:数组变为 ['a', '2', 'b', '2', 'c', '3'],新长度为 6。

示例 2

输入chars = ['a']

步骤解析

  1. read = 0current = 'a',计数为 1。

    • 写入 'a'write = 1

    • 因为计数为 1,不写入数字。

    • read 移动到 1,循环结束。

输出:数组仍为 ['a'],新长度为 1。

示例 3

输入chars = ['a', 'b', 'b', 'b', 'b', 'b', 'b', 'b', 'b', 'b', 'b', 'b', 'b']

步骤解析

  1. read = 0current = 'a',计数为 1。

    • 写入 'a'write = 1

    • read 移动到 1。

  2. read = 1current = 'b',计数为 12。

    • 写入 'b'write = 2

    • 写入 '1''2'write = 4

    • read 移动到 13,循环结束。

输出:数组变为 ['a', 'b', '1', '2'],新长度为 4。

八、常见问题解答

Q1:为什么选择双指针法?

双指针法能够高效地在原地完成数组的压缩。它通过两个指针分别负责读取和写入操作,避免了使用额外的存储空间。这种方法在时间复杂度和空间复杂度上都具有显著优势,特别适合处理这类原地修改的问题。

Q2:如何处理计数大于 9 的情况?

当计数大于 9 时,将计数转换为字符串,然后逐个字符写入数组。例如,计数为 12 时,写入 '1''2'。这种方法确保了所有计数都能正确表示,无论其大小如何。

Q3:算法是否适用于所有长度的数组?

是的。该算法适用于所有长度的数组,包括空数组、单个字符数组以及包含多个重复字符组的数组。在代码开头,我们对空数组进行了特殊处理,返回 0。对于其他情况,算法都能正确处理。

九、总结

字符数组压缩算法是一个典型的原地操作问题,通过双指针法能够在 O(n) 时间复杂度和 O(1) 空间复杂度下完成任务。该算法的核心在于高效地识别连续重复字符组,并将它们转换为压缩形式。通过详细解析和示例分析,我们希望读者能够深入理解这一算法的原理和实现细节。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/76280.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

alertManager部署安装、告警规则配置详解及告警消息推送

​ java接受告警请求RestController RequestMapping("/alert") Slf4j public class TestApi {private static final DateTimeFormatter FORMATTER DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");RequestMappingpublic void sendTemplate(HttpServl…

数据库勒索病毒威胁升级:企业数据安全防线如何用安当RDM组件重构

摘要&#xff1a;2025年Q1全球数据库勒索攻击量同比激增101.8%&#xff0c;Cl0p、Akira等团伙通过边缘设备漏洞渗透企业核心系统&#xff0c;制造业、金融业等关键领域面临数据加密与业务停摆双重危机。本文深度解析勒索病毒对数据库的五大毁灭性影响&#xff0c;结合安当RDM防…

thanos sidecar和receive区别?

Thanos Sidecar 和 Thanos Receive 是 Thanos 生态系统中两个关键组件&#xff0c;但它们在架构中的作用和功能上有明显的区别。以下是它们的主要区别&#xff1a; 1. Thanos Sidecar 功能&#xff1a; 与 Prometheus 集成&#xff1a; Sidecar 是一个部署在每个 Prometheus…

Unity入门笔记(缘更)

内容来源SiKi学院的Luna’s Fantasy 文章目录 一、基础知识1.准备2.基础知识1.层级(Layer)2.轴心点3.预制体(Prefab)4.刚体组件(Rigidbody)5.碰撞器组件(BoxCollider) 二、代码1.移动 一、基础知识 1.准备 Unity安装&#xff1a; https://unity.cn 2.基础知识 1.层级(Layer…

使用VHD虚拟磁盘安装双系统,避免磁盘分区

前言 很多时候&#xff0c;我们对现在的操作系统不满意,就想要自己安装一个双系统 但是安装双系统又涉及到硬盘分区,非常复杂,容易造成数据问题 虚拟机的话有经常用的不爽,这里其实有一个介于虚拟机和双系统之间的解决方法,就是使用虚拟硬盘文件安装系统. 相当于系统在机上…

ARINC818协议(五)

1.R_CTL,设置固定的0x44即可 2.Dest_ID:目的地D_ID,如果不需要目的地址&#xff0c;就设置为0&#xff1b;ADVB协议支持 多个视频目的地址&#xff0c;广播通信; 3.cs_ctl在FC-AV上不用 4.source_ID:S_ID [23:0]包含源实体的端口的地址标识&#xff1b;不用就设置为0. ADVB允许…

鸿蒙开发对于RelativeContainer高度设置‘auto‘后还是没有自适应问题的解决方案

RelativeContainer设置高度为自适应‘auto’没用生效&#xff0c;查看了官方文档(文档中心)也没用给出明确的答案。只说了不能把锚点设置成父组件锚点&#xff08;__container__&#xff09;。也尝试了使用guidline来替换父组件锚点&#xff0c;还是没能自适应高度。 后来尝试让…

k8s教程3:Kubernetes应用的部署和管理

学习目标 理解Kubernetes中应用部署的基本概念和方法掌握Deployment、ReplicaSet、StatefulSet、DaemonSet、Job与CronJob等控制器的使用了解Helm作为Kubernetes的包管理工具的基本使用通过实际示例学习应用的部署、更新与管理 Kubernetes提供了一套强大而灵活的机制&#xff…

通过特定协议拉起 electron 应用

在 Android 通过 sheme 协议可以拉起其他应用。 electron 应用也可以通过类似特定协议被拉起。 在同时有 web、客户端的应用里&#xff0c;可以通过这种方式在 web 拉起客户端。 支持拉起客户端 const PROTOCOL xxxif (process.defaultApp) {// 这里是开发环境&#xff0c;有…

算法备案的审核标准是什么?

随着《互联网信息服务算法推荐管理规定》等法规的出台&#xff0c;算法备案成为了强制性备案&#xff0c;是产品合规上线的必要条件之一。本篇内容将从企业视角出发&#xff0c;分析算法备案的常见问题&#xff0c;意在对有备案需求的小伙伴们有所帮助。 一、谁需要做算法备案…

回顾与动机 - 为什么我们需要 Transformer

在接下来的旅程中,我们将一起探索深度学习领域最重要、最具影响力的模型架构之一——Transformer。从它的基本原理出发,逐步深入,最终能够亲手实现一个文本生成模型。 本系列教程假设你已经具备一定的深度学习基础,了解神经网络、损失函数、优化器等基本概念,并且熟悉 Py…

探索 Higress:下一代云原生 API 网关

引言 在云原生时代&#xff0c;API 网关作为连接客户端与后端服务的桥梁&#xff0c;扮演着至关重要的角色。Higress 是一款由阿里巴巴开发的先进云原生 API 网关&#xff0c;基于开源的 Istio 和 Envoy 构建。它通过将流量网关、微服务网关和安全网关三者高度集成&#xff0c…

Spring Boot 整合 DeepSeek 实现AI对话 (保姆及教程)

文章目录 文章目录 前言 一、创建 spring boot 工程 二、申请key 三、修改配置文件 application.properties 四、编写控制器&#xff08;controller&#xff09; 五、运行调试 前言 提示&#xff1a;随着人工智能的不断发展&#xff0c;ai这门技术也越来越重要&#xff0c;很多…

前端资源加载失败后重试加载(CSS,JS等引用资源)

前端资源加载失败后的重试 .前端引用资源时出现了资源加载失败(这里针对的是路径引用异常或者url解析错误时) 解决这个问题首先要明确一下几个步骤 1.什么情况或者什么时候重试 2.如何重试 3.重试过程中的边界处理 这里引入里三个测试脚本&#xff0c;分别加载里三个不同的脚…

无刷电机槽数相同、转子极数不同的核心区别

一、基础原理差异 无刷电机的核心参数: 槽数(定子槽数,记为 ( Z )):定子铁芯上的绕组槽数量,决定绕组布局。极数(转子磁极数,记为 ( 2p )):转子上的永磁体磁极对数(总极数为 ( 2p ),如 ( p=4 ) 表示 8 极)。核心关系:槽极配合(( Z/2p ))决定电机电磁结构,相同…

6.Rust+Axum:打造高效 WebSocket 实时通信聊天室

摘要 本文详细介绍 RustAxum 在 WebSocket 实时通信开发中的应用&#xff0c;包括双向通信、状态管理等&#xff0c;实践构建聊天室应用。 一、引言 在当今的 Web 应用开发中&#xff0c;实时通信变得越来越重要。WebSocket 作为一种在单个 TCP 连接上进行全双工通信的协议&…

clickhouse数据导出导入

clickhouse数据导出导入 CSV格式导出为csv格式导入为csv格式 JSON格式导出为json格式导入为json格式 SQL格式导出为SQL CSV格式 导出为csv格式 # 不带表头 clickhouse-client -h 127.0.0.1 --database"db" --query"select * from db.test_table FORMAT CSV&qu…

人脸扫描黑科技:多相机人脸扫描设备,打造你的专属数字分身

随着科技的迅猛发展&#xff0c;人脸扫描这个词已经并不陌生&#xff0c;通过人脸扫描设备制作超写实人脸可以为影视制作打造逼真角色、提升游戏沉浸感&#xff0c;还能助力教育机构等领域生产数字人以丰富教学资源&#xff0c;还在安防、身份识别等领域发挥关键作用&#xff0…

学习型组织与系统思考

真正的学习型组织不是只关注个人的学习&#xff0c;而是关注整个系统的学习。—彼得圣吉 在这两年里&#xff0c;越来越多的企业开始询问是否可以将系统思考的内容内化给自己的内训师&#xff0c;进而在公司内部进行教学。我非常理解企业这样做的动机&#xff0c;毕竟内部讲师…

gl-matrix 库简介

gl-matrix 库简介 gl-matrix 是一个高性能的 JavaScript 矩阵和向量库&#xff0c;专门为 WebGL 和其他 3D 图形应用设计。它提供了处理 2D、3D 和 4D 向量以及矩阵运算的高效方法。 主要特性 高性能&#xff1a;经过高度优化&#xff0c;执行速度快轻量级&#xff1a;体积小…