面试 | 双法妙解压缩字符串【遍历统计 + 双指针】

在这里插入图片描述

一、题目描述

原题传送门

在这里插入图片描述

二、思路分析

首先我们来分析一下解决本题所需要的思路

  • 题目的意思很简单,就是统计原本的字符串中的每个字符出现的次数,然后以【字符,出现的次数】这样的结构来字符串,以起到一个压缩的效果,那么对于这样的结构,详细很多力友都会想到map这个键值对的结构,但是认真查看题目本身的话却发现我们不可以用这种结构

输入:“aabcccccaaa”
输出:“a2b1c5a3”

  • 我们以第一个为例,从左往右分别是2个a、1个b、5个c和3个a,此时若我们通过map去做遍历的话,开头的【a】和结尾的【a】就会合并在一起变为5个a,那么最后在拼接字符串的时候就会出现问题,所以map的这种结构我们是不能使用
unordered_map<char, int> count
for(char c : S)
{count[c]++;
}

接下去就来看看下面的两种做法👇

Way1 : 遍历统计

  • 首先是第一种方法,既然题目给到了一个字符串,那我们就去遍历它并进行一个统计,那既然是统计某个字符出现了多少次的话,就需要有一个字符变量去记录,这里我们拿ch首先保存下第一个字符,后面遍历的时候再做更新
char ch = S[0]; 
  • 当然,我们还需要一个计数器cnt
  • 接下去这段逻辑就是在遍历这个字符串的同时去统计里面的每个字符出现了多少次,如果当前所遍历的字符和ch是相同的话,计数器cnt就去做一个累加,如果不相同的话,我们就可以去拼接字符串了,ch即为当前所需要统计的字符,cnt即为这个字符所出现的次数,不过我们要使用到C++里的一个库函数叫做 to_string ,不清楚的力友可以先去了解一下
  • 当这个字符拼接完后,我们就要去统计下一个字符了,这个时候就要更新【ch】和【cnt】了
string compressStr = "";
for(int i = 1;i < sz; ++i)
{if(S[i] == ch){cnt++;      // 如果当前所遍历到的字符与需统计字符相同的话,则计数器 + 1}else{compressStr += ch + to_string(cnt);ch = S[i];cnt = 1;}
}
  • 最后呢,在遍历结束之后我们还需要在去做一个拼接,因此当最后遍历最后跳出循环并没有把最后一个字符给拼接上去
// 最后在遍历结束后添加最后一个字符的统计结果
compressStr += ch + to_string(cnt);
  • 当字符拼接完后,就将其返回,但是题目中说到如果压缩之后的字符串长度比原字符串还长的话,就返回原先的串
return compressStr.size() >= sz ? S : compressStr;
  • 不过这里还有隐藏的一个测试案例,我们应该要考虑到压缩后的字符串和原串相等的情况,也是需要返回原串的

1.jpg

代码展示:

class Solution {
public:string compressString(string S) {int sz = S.size();if(sz == 0)return S;   // 如果是空串的话,则返回自己本身string compressStr = "";int cnt = 1;        // 统计每一个字符个数char ch = S[0];     // 记录当前所要统计的字符// 循环直接从1开始即可,第一个字符一定算一个for(int i = 1;i < sz; ++i){if(S[i] == ch){cnt++;      // 如果当前所遍历到的字符与需统计字符相同的话,则计数器 + 1}else{compressStr += ch + to_string(cnt);ch = S[i];cnt = 1;}}// 最后在遍历结束后添加最后一个字符的统计结果compressStr += ch + to_string(cnt);// 三目运算符:若是压缩后的字符串长度 >= 原字符串的长度,则返回原串return compressStr.size() >= sz ? S : compressStr;}
};

最后展示一下AC后的结果

2.jpg

Way2 : 双指针

接下去要介绍的就是另一种解法,此解法来自 K神 很是巧妙,力友可以学习一下这种解法

图解分析

总体思路概述:

  • 这种方法很巧妙地利用了双指针,首先让指针ij指向当前字符串的首位,然后让指针j不断地先后遍历,比较 s[i] 和 s[j] 的的字符是否相同,若是相同的话则让j继续后移,直到二者不相同为止。
  • 此时我们便去计算当前的这个字符出现了多少次,使用【指针 - 指针】的办法算出来。然后再去统计下一个字符的位置,此时让i移动到j的位置来即可,因为二者不相同后j一定指向了一个新的字符
  • 然后继续重复上面的逻辑就可以了,直到这个字符串遍历完之后也就压缩完了

然后我们就根据 示例1 来分布细述一下

  • 首先双指针都指向字符串的首位置

3.jpg

  • 然后因为【a = a】,所以j++i不动

4.jpg

  • 继续还是一样,【a = a】,j++

5.jpg

  • 那么接下来【a != b】,遇到不相等了,此时我们就需要去统计当前这个字符所出现的此处,使用j - i就可以计算得到,那么此时就可以将这结果追加到压缩后的字符串中了。接下去便是更新指针i的位置到j这里来,进行下一个字符串个数的统计

6.jpg

  • 接下去就开始统计字符b出现的个数了,但是呢在比较了一次够就不相等了,于是立马就得出这个字符只出现了一次,步骤和上述一样

7.jpg

  • 接下去就是字符【c】了,利用j - i可以算出其出现了5次,继续拼接到压缩串compressStr中

8.jpg

  • 最后当这个串遍历完之后,因为没有再进入循环了,所以我们在最后还要拼接一次

9.jpg

  • 最后的话当我们要返回结果的时候还需要比较一下压缩后的串和原串的长度,只有当压缩后的串来得短的时候才可返回,否则一律返回原串

10.jpg

代码展示:

class Solution {
public:string compressString(string S) {if(S.size() == 0)   return S;int i = 0, j = 0;string compressStr = "";while(j < S.size()){// 持续比较双指针上的位置,直到不相同为为止while(S[i] == S[j]){j++;}compressStr += S[i];compressStr += to_string(j - i);i = j;      // i换位到新字符的位置}return compressStr.size() >= S.size() ? S : compressStr; }
};

最后从提交的结果就可以看出效率提升了不少 ↑

11.jpg

认真看完上面这个例子后,其他的示例相信你一定也能推敲出来🌹

在这里插入图片描述

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

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

相关文章

网络安全:密码学基本理论.

网络安全&#xff1a;密码学基本理论. 密码学是研究编制密码和破译密码的技术科学。研究密码变化的客观规律&#xff0c;应用于编制密码以保守通信秘密的&#xff0c;称为编码学&#xff1b;应用于破译密码以获取通信情报的&#xff0c;称为破译学&#xff0c;总称密码学. 目录…

ffmpeg离线安装ffmpeg-4.1.4-amd64-static.tar.xz

1.下载离线包 John Van Sickle - FFmpeg Static Builds 找历史版本&#xff1a;Index of /ffmpeg/old-releases 我选择是的4.1.4版本 2.解压 tar -xvJf ffmpeg-4.1.4-amd64-static.tar.xz 3.移动文件到opt目录下 4.添加全局链接 ln -s /opt/ffmpeg-4.1.4-amd64-static/ffm…

在macOS、Windows上使用VSCode + SSH实现远程Matplotlib图形显示

简介 在 macOS 上使用 VSCode SSH 环境来显示 Matplotlib 绘制的图形需要进行一些配置。因为默认情况下&#xff0c; Matplotlib 的图形是无法显示在远程计算机的桌面上的。您可以通过设置 Matplotlib 使用不同的后端(backend)来实现将图形显示在本地计算机上。 第一步&…

Docker安装Nacos2.0.2

docker拉取镜像 docker pull nacos/nacos-server:2.0.2查看镜像 docker images创建容器和运行 docker run -e JAVA_OPTS"-Xms256m -Xmx256m" -e MODEstandalone -e PREFER_HOST_MODEhostname -p 8848:8848 --privilegedtrue --restartalways --name nacos -d naco…

小程序制作教程

步骤一&#xff1a;规划和设计 在开始制作微信小程序之前&#xff0c;首先需要规划和设计您的小程序。确定您想要提供的服务或功能&#xff0c;并考虑用户体验和界面设计。绘制草图和构思完整的页面布局&#xff0c;这将使您更好地理解小程序结构和功能。 步骤二&#xff1a;…

Python Selenium设计模式及代码实现

前言 本文就python selenium自动化测试实践中所需要的POM设计模式进行分享&#xff0c;以便大家在实践中对POM的特点、应用场景和核心思想有一定的理解和掌握。 为什么要用POM 基于python selenium2开始UI级自动化测试并不是多么艰巨的任务。**只需要定位到元素&#xff0c;…

公网访问的Linux CentOS本地Web站点搭建指南

文章目录 前言1. 本地搭建web站点2. 测试局域网访问3. 公开本地web网站3.1 安装cpolar内网穿透3.2 创建http隧道&#xff0c;指向本地80端口3.3 配置后台服务 4. 配置固定二级子域名5. 测试使用固定二级子域名访问本地web站点 前言 在web项目中,部署的web站点需要被外部访问,则…

基于SPDK-vhost的云原生Kubevirt虚拟化存储IO的优化方案

摘要 本文主要介绍针对云原生kubernetes虚拟化IO的应用场景&#xff0c;在Kubevirt中引入SPDK-vhost的支持&#xff0c;来加速虚机中IO存储性能。同时基于Intel开源的Workload Service Framework[1]平台集成部署一套端到端虚拟化IO的应用场景做基本的性能对比测试。 云原生Kube…

人员定位安全管控系统:提升安全管理水平的智能解决方案

在当今社会&#xff0c;人员安全管理成为各行各业关注的焦点。为了保障人员的安全和提高管理效率&#xff0c;人员定位安全管控系统应运而生。 人员定位安全管控系统采用多种定位技术来实现对人员位置的准确定位&#xff0c;如GPS&#xff08;全球定位系统&#xff09;、Wi-Fi…

超低输入电压升压电路解决方案

便携式产品一般都采用电池供电&#xff0c;而因为成本和体积方面的考虑&#xff0c;在设计上有减少使用电池数量及体积的趋势。另外&#xff0c;亦因全球能源问题&#xff0c;各种各类的电池使用已备受关注了。当中包括太阳能电池及燃料电池。 而这样就会影响到电源电压比设备所…

Flask的send file和send_from_directory的区别

可以自行查看flask 文档。 send file高效&#xff1b; send from directory安全&#xff0c;且适用于静态资源交互。 都是实现相同的功能的。 send_file send_from_directory

直播回顾 | SDS 容灾方案,让制品数据更安全

7 月 18 日&#xff0c;腾讯云 CODING 与 XSKY星辰天合联合举办了主题为“SDS 容灾方案&#xff0c;让制品数据更安全”的线上研讨会。 来自腾讯云 CODING 的高级解决方案架构师陈钧桐和 XSKY星辰天合金融行业解决方案专家战策&#xff0c;分享了制品管理的困境与需求、腾讯云…

blender 基础材质篇

材质展示 材质背景介绍 什么是PBR&#xff1f; PBR 全称为 Physically Based Rendering&#xff0c;译为基于物理属性的引擎渲染&#xff0c;也就是说会把物质的颜色、粗糙度、高光属性等进行分别处理&#xff0c;使物质体现出更真实的感觉&#xff1b; 什么是BRDF&#xff…

前端工程师的岗位职责(合集)

篇一 岗位职责&#xff1a; 1、负责网站前端开发&#xff0c;实现产品的页面交互及功能实现; 2、与程序开发人员紧密合作&#xff0c;制作前端及后端程序接口标准; 3、完成产品的设计、开发、测试、修改bug等工作&#xff0c;包括业务需求的沟通&#xff0c;功能模块详细设计…

背包问题(一)

题目一 01背包 有 N 件物品和一个容量是 V的背包。每件物品只能使用一次。 第 i 件物品的体积是 vi&#xff0c;价值是 wi。 求解将哪些物品装入背包&#xff0c;可使这些物品的总体积不超过背包容量&#xff0c;且总价值最大。 输出最大价值。 输入格式 第一行两个整数…

力扣C++|一题多解之数学题专场(2)

目录 50. Pow(x, n) 60. 排列序列 66. 加一 67. 二进制求和 69. x 的平方根 50. Pow(x, n) 实现 pow(x,n)&#xff0c;即计算 x 的 n 次幂函数&#xff08;即x^n&#xff09;。 示例 1&#xff1a; 输入&#xff1a;x 2.00000, n 10 输出&#xff1a;1024.00000 示例…

听GPT 讲K8s源代码--pkg(八)

k8s项目中 pkg/kubelet/envvars&#xff0c;pkg/kubelet/events&#xff0c;pkg/kubelet/eviction&#xff0c;pkg/kubelet/images&#xff0c;pkg/kubelet/kubeletconfig这些目录都是 kubelet 组件的不同功能模块所在的代码目录。 pkg/kubelet/envvars 目录中包含了与容器运行…

JVM之内存与垃圾回收篇3

文章目录 8 垃圾回收8.1 基本理论8.1.1 对象的finalization机制8.1.2 理解System.gc8.1.3 内存溢出和内存泄漏8.1.4 Stop The World8.1.5 安全点和安全区域8.1.6 Java中的引用 8.2 垃圾回收算法8.2.1 引用计数法8.2.2 可达性分析8.2.2.1 使用MAT查看GC Roots8.2.2.2 使用JProfi…

整车总线系列——FlexRay 四

整车总线系列——FlexRay 四 我是穿拖鞋的汉子&#xff0c;魔都中坚持长期主义的汽车电子工程师。 老规矩&#xff0c;分享一段喜欢的文字&#xff0c;避免自己成为高知识低文化的工程师&#xff1a; 没有人关注你。也无需有人关注你。你必须承认自己的价值&#xff0c;你不能…

Spring使用注解存储Bean对象

文章目录 一. 配置扫描路径二. 使用注解储存Bean对象1. 使用五大类注解储存Bean2. 为什么要有五大类注解&#xff1f;3.4有关获取Bean参数的命名规则 三. 使用方法注解储存Bean对象1. 方法注解储存对象的用法2. Bean的重命名 在前一篇博客中&#xff08; Spring项目创建与Bean…