【感悟《剑指offer》典型编程题的极练之路】02字符串篇!

                                                                                个人主页:秋风起,再归来~

                                                               文章所属专栏:《剑指offer》典型编程题的极练之路                        ​​​​​​                                     

                                                                        个人格言:悟已往之不谏,知来者犹可追

                                                                                        克心守己,律己则安!

目录

一、C/C++中字符数组与字符常量

二、面试题(替换空格)

2.1时间复杂度为 Q(㎡)的解法,不足以拿到 Offer

2.2 创建新的数组解题

2.3时间与空间复杂度分别为O(N)和O(1)的算法!(搞定offer就靠它了!)

​编辑

三. 完结散花


一、C/C++中字符数组与字符常量

为了节省空间,C/C++把常量字符串放到单独的一个内存区域。当几个指针赋值给相同的常量字符串时,它们实际上会指向相同的内存地址。但用常量内存初始化数组,情况却有所不同下面通过一个面试题来学习这一知识点。

#define _CRT_SECURE_NO_WARNINGS#include<stdio.h>int main()
{char* s1 = "abcdef";char* s2 = "abcdef";char* s3[] = { "abcdef" };char* s4[] = { "abcdef" };if (s1 == s2){printf("s1与s2相等\n");}else{printf("s1与s2不相等\n");}if (s3 == s4){printf("s3与s4相等\n");}else{printf("s3与s4不相等\n");}return 0;
}

如果我们运行下面代码的话结果是什么呢?

为什么呢?

我们先通过调试来理解一下?

我们可以看到s1和s2的值相等,说明它们指向了同一块内存地址!

反之,s3与s4所指向的空间则是不同的!

那这又是为什么呢?

首先我们知道常量是不能被改变的,当内存中开辟了一片空间存放一个字符串常量并且有一个指针指向它,有朝一日,我们不小心又创建了一个相同的常量字符串并且用另一个指针指向它,这时内存并不会再创建一片空间来存放这个常量字符串,而是直接让另一个指针指向向前开辟的空间。

那为什么字符数组却不同呢,那是因为字符数组是可修改的,如果它们指向同一块空间,有朝一日,我想修改这个字符数组,那另一个字符数组必然也会被修改!

二、面试题(替换空格)

描述:

请实现一个函数,将一个字符串s中的每个空格替换成“%20”。

例如,当字符串为We Are Happy.则经过替换之后的字符串为We%20Are%20Happy。

数据范围:0≤en(s)≤1000 。保证字符串中的字符为大写英文字母、小写英文字母和空格中的一种。

题目链接:点击进入

在网络编程中,如果URL参数中含有特殊字符,如空格“#”等,则可能导致服务器端无法获得正确的参数值。我们需要将这些特殊符号转换成服务器可识别的字符。转换的规则是在%后面跟上ASCLL码的两位十六进制的表示。比如空格的ASCLL码是32即十六进制的0x20,因此空格被替换为%20,再比如#的ASCLL为35,即十六进制的0x23,他在URL中被替换为%23。

看到这个题目,我们首先应该想到的是原来一个空格字符,替换之后变成"%、"2和0这3个字符,因此字符串会变长。如果是在原来的字符串上进行替换,就有可能覆盖修改在该字符串后面的内存。如果是创建新的字符串并在新的字符串上进行替换,那么我们可以自己分配足够多的内存。由于有两种不同的解决方案,我们应该向面试官问清楚,让他明确告诉我们他的需求。假设面试官让我们在原来的字符串上进行替换,并且保证输入的字符串后面有足够多的空余内存。

2.1时间复杂度为 Q(㎡)的解法,不足以拿到 Offer


现在我们考虑怎么执行替换操作。最直观的做法是从头到尾扫描字符每次碰到空格字符的时候进行替换。由于是把1个字符替换成3个字串,符,我们必须要把空格后面所有的字符都后移2字节,否则就有两个字符被覆盖了。
举个例子,我们从头到尾把"We are happy,"中的每个空格替换成"%20"。为了形象起见,我们可以用一个表格来表示字符串,表格中的每个格子表示一个字符,如图 所示。

注:(a)字符串"Weare happy."。(b)把字符串中的第一个空格替换成"%20”。黄色背景表示需要移动的字符。(c)把字符串中的第二个空格替换成”%20"。黄色背景表示需要移动一次的字符,红色色背景表示需要移动两次的字符。
我们替换第一个空格,这个字符串变成图(b)中的内容,表格中黄色背景的格子表示需要进行移动的区域。接着我们替换第二个空格,替换之后的内容如图2.3(c)所示。同时,我们注意到用红色背景标注的happy”部分被移动了两次。
假设字符串的长度是n。对每个空格字符,需要移动后面O(n)个字符:因此对于含有 0(n)个空格字符的字符串而言,总的时间效率是 O(n’)。
当我们把这种思路阐述给面试官后,他不会就此满意,他将让我们寻找更快的方法。在前面的分析中,我们发现数组中很多字符都移动了很多次,能不能减少移动次数呢?答案是肯定的。我们换一种思路,把从前向后替换改成从后向前替换。

2.2 创建新的数组解题

结合代码注释和图解就很容易理解这种算法了~

char* replaceSpace(char* s ) {// write code hereif(s==NULL)return NULL;//先记录与字符串的大小int len=strlen(s);//记录空格的数量int count=0;int i=0;while(s[i]!='\0'){if(s[i]==' ')count++;i++;}//创建一个新的字符数组char* newStr=(char*)malloc(sizeof(char)*(len+count*2));//将原字符串的内容拷贝进新的字符数组,遇到空格就替换int j=0;int end=0;while(s[end]!='\0'){if(s[end]==' '){newStr[j++]='%';newStr[j++]='2';newStr[j++]='0';}else{newStr[j++]=s[end];}end++;}return newStr;
}

2.3时间与空间复杂度分别为O(N)和O(1)的算法!(搞定offer就靠它了!)

char* replaceSpace(char* s ) {// write code hereint end=0;if(s==NULL)return NULL;int count=0;//记录空格的数量while(s[end]!='\0')//找到字符串的末尾{if(s[end]==' '){count++;}end++;}//找到替换过后字符串的末尾int newEnd=end+2*count;//从后往前移动字符并且替换空格while(end>=0&&end<newEnd){if(s[end]==' '){s[newEnd--]='0';s[newEnd--]='2';s[newEnd--]='%';}else{s[newEnd--]=s[end];}end--;}return s;
}

这个算法的核心在于如何高效地将字符串中的空格替换为"%20",同时保证不覆盖或遗漏任何字符。算法的原理可以概括为以下几个步骤:

步骤一:计算空格数量
首先,我们需要遍历一遍输入的字符串,统计其中空格的数量。这是因为每个空格都将被替换为三个字符('%'、'2' 和 '0'),所以我们需要知道有多少个空格,以便计算替换后字符串的总长度。

步骤二:计算新字符串长度
接下来,我们根据原始字符串的长度和空格的数量,计算出替换后字符串的总长度。由于每个空格替换为三个字符,所以新字符串的长度将是原始长度加上空格数量乘以2(因为每个空格增加了两个字符)。

步骤三:从后向前遍历并替换
然后,我们从字符串的末尾开始,向前遍历原始字符串。这样做的目的是为了避免在替换过程中覆盖还未处理的字符。对于每个字符,我们检查它是否是空格。如果是空格,我们就将其替换为"%20",并更新新字符串的索引位置。如果不是空格,我们则直接将字符复制到新字符串的对应位置,并更新索引。

这个算法的关键在于利用了字符串的末尾空间,从后向前进行替换操作,避免了额外的内存分配和字符串拷贝。它只需要一次遍历就能完成替换操作,时间复杂度是O(n),其中n是字符串的长度。因此,这个算法是高效且实用的。

此外,值得注意的是,这个算法直接修改了输入的字符串,而不是创建了一个新的字符串。这意味着在调用这个函数之后,原始的字符串已经被修改。如果原始字符串不应该被修改,那么在调用这个函数之前,你需要先复制一份原始字符串。

注:这个算法假设输入的字符串有足够的空间来容纳替换后的结果。如果原始字符串的空间不足以容纳替换后的结果,那么这个算法可能会导致缓冲区溢出。因此,在实际使用时,你需要确保输入的字符串有足够的空间,或者在调用这个函数之前,先分配一个足够大的新字符串来存放替换后的结果。

值得一提的是:这种算法在牛客上并不能通过全部的测试用例,我估计是后台调用函数时传递的字符数组并没有足够大的空间,导致数组越界访问了~

一些感悟:在面试的过程中,我们也可以和前面的分析一样画一两个示意图解释自己的思路,这样既可以帮助我们厘清思路,也可以使我们和面试官交流更加的高效!

三. 完结散花

好了,这期的分享到这里就结束了~

如果这篇博客对你有帮助的话,可以用你们的小手指点一个免费的赞并收藏起来哟~

如果期待博主下期内容的话,可以点点关注,避免找不到我了呢~

我们下期不见不散~~

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

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

相关文章

python入门题:输入输出练习

以下是Python基础语法的练习&#xff0c;项目要求和代码如下&#xff1a; """ 例3&#xff1a;小精灵&#xff1a;你好&#xff0c;欢迎古灵阁&#xff0c;请问您需要帮助吗&#xff1f;需要or不需要&#xff1f; 你&#xff1a;需要 小精灵&#xff1a;请问你需…

[ZKP] Freivalds’ Algorithm

Freivalds’ Algorithm Freivalds, Rusins. “Probabilistic Machines Can Use Less Running Time.” IFIP congress. Vol. 839. 1977. Problem Statement Suppose we are given as input two n n n \times n nn matrices A A A and B B B over F p \mathbb{F}_p Fp​, w…

代码随想录刷题笔记 Day 59 | 两个字符串的删除操作 No.583 | 编辑距离 No.72

文章目录 Day 5901. 两个字符串的删除操作&#xff08;No. 583&#xff09;<1> 题目<2> 题解<3> 代码 02. 编辑距离&#xff08;No. 72&#xff09;<1> 题目<2> 题解<3> 代码 Day 59 01. 两个字符串的删除操作&#xff08;No. 583&#x…

MFC(二)集成基础控件

目录 OnCreateCStatic【标签&#xff0c;图片】CEdit【文本框&#xff0c;密码框&#xff0c;数值框&#xff0c;文本区】CButton【按钮&#xff0c;单选按钮&#xff0c;多选按钮】CComboBox【下拉列表&#xff0c;列表】CSliderCtrl【滑动条】CListCtrl【表格】CAnimateCtrl【…

【Java程序设计】【C00383】基于(JavaWeb)Springboot的水产养殖系统(有论文)

【C00383】基于&#xff08;JavaWeb&#xff09;Springboot的水产养殖系统&#xff08;有论文&#xff09; 项目简介项目获取开发环境项目技术运行截图 博主介绍&#xff1a;java高级开发&#xff0c;从事互联网行业六年&#xff0c;已经做了六年的毕业设计程序开发&#xff0c…

Sentinel源码解析

核心源码都在客户端,服务端只是个Dashboard!!! 在服务端配置好规则后,服务端会把规则推到客户端里去【存在客户端内存里】 服务端记录客户端对外提供的一些接口 客户端引用了依赖并启动后,会定时把自己的信息注册到Sentinel服务端去,并且定时发信息保持心跳 主线 注解…

编程出现bug?怎么用Python打印异常

在 Python 编程中&#xff0c;异常是指程序执行过程中出现的错误或异常情况。当程序遇到异常时&#xff0c;为了更好地调试和定位问题&#xff0c;我们需要打印异常信息。本文将详细介绍如何在 Python 中打印异常&#xff0c;并提供一些示例和注意事项。 一、try-except 语句捕…

【数学】 【分数】 【字符串】972. 相等的有理数

本文涉及知识点 数学 分数 字符串 LeetCode972. 相等的有理数 给定两个字符串 s 和 t &#xff0c;每个字符串代表一个非负有理数&#xff0c;只有当它们表示相同的数字时才返回 true 。字符串中可以使用括号来表示有理数的重复部分。 有理数 最多可以用三个部分来表示&…

【保姆级教程】使用SeaTunnel同步Kafka的数据到ClickHouse

1.Apache SeaTunnel依赖地址 2.SeaTunnel官网的Source/Sink模板 3.SeaTunnel的GitHub地址 在官网下载安装包之后&#xff0c;&#xff08;注意&#xff1a;别下载apache-seatunnel-incubating-2.1.0-bin.tar.gz版本&#xff0c;依赖和功能都没有。)要使用apache-seatunnel-2.3…

什么是V R美术馆|V R互动体验店加盟|虚拟现实元宇宙

VR美术馆是利用虚拟现实&#xff08;VR&#xff09;技术构建的数字化美术馆&#xff0c;通过虚拟展厅和虚拟展览等形式展示艺术作品、举办艺术展览&#xff0c;为用户提供一种沉浸式的艺术体验。用户可以通过穿戴VR头显等设备&#xff0c;在虚拟环境中自由浏览各种艺术作品&…

SpringCloud学习笔记二:服务间调用

微服务中&#xff0c;很多服务系统都在独立的进程中运行&#xff0c;通过各个服务系统之间的协作来实现一个大项目的所有业务功能。服务系统间 使用多种跨进程的方式进行通信协作&#xff0c;而RESTful风格的网络请求是最为常见的交互方式之一。 spring cloud提供的方式&#…

matplotlib中的颜色表示方法

matplotlib中的颜色表示方法 1.RGB或RGBA格式 格式示例以一个3元素或4元素的tuple来表示颜色&#xff0c;每个元素取值范围是[0,1](0.1,0.2,0.5) (0.1,0.2,0.5,0.3)大小写不敏感的16进制表示法#0F0F0F等价于#0x0f0f0f等价于(15/255,15/255,15/255)带透明度的#0f0f0f80简短的…