【感悟《剑指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.rhkb.cn/news/291619.html

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

相关文章

2013年认证杯SPSSPRO杯数学建模B题(第一阶段)流行音乐发展简史全过程文档及程序

2013年认证杯SPSSPRO杯数学建模 B题 流行音乐发展简史 原题再现&#xff1a; 随着互联网的发展&#xff0c;流行音乐的主要传播媒介从传统的电台和唱片逐渐过渡到网络下载和网络电台等。网络电台需要根据收听者的已知喜好&#xff0c;自动推荐并播放其它音乐。由于每个人喜好…

C语言goto语句介绍

在C语言中&#xff0c;goto语句是一种流程控制语句&#xff0c;用于无条件地转移到程序中的特定标签位置。尽管goto语句在编程中具有一定的争议&#xff0c;但在某些情况下&#xff0c;它可以提供一种简单有效的解决方案。本文将深入介绍C语言中的goto语句&#xff0c;包括其基…

5个免费的3D钣金CAD软件

如果你正在设计简单的折叠钣金零件&#xff0c;则只需设计一些具有圆角半径的法兰&#xff1a;一个简单的钣金模块。 首先&#xff0c;你可以采用老式方式绘图并以 2D 方式完成所有操作。 许多传统制造商仍在使用 2D DWG 和 DXF 图纸。 因此&#xff0c;你很有可能只需快速起草…

18.8K星开源免费的跨平台密码管理器:KeePassXC

KeePassXC&#xff1a;您的跨平台密码守护神&#xff0c;安全存储&#xff0c;随心所欲&#xff0c;无论何处皆可信手拈来! - 精选真开源&#xff0c;释放新价值。 概览 当你面临一堆应用需要填写各种各样的密码的时候、当你需要记忆各种各样的密码或是需要保存保密文件或私密…

【win10 win11添加右键】git bash

打开注册表编辑器。 按下Win键 R&#xff0c;然后输入”regedit”并按下回车键来打开注册表编辑器。计算机\HKEY_CLASSES_ROOT\Directory\Background\shell\git_bash\command2. 导航到注册表路径&#xff1a;依次展开”HKEY_CLASSES_ROOT\Directory\Background\shell”。右键…

怎样完成票据证件的关键信息抽取任务

一个不知名大学生&#xff0c;江湖人称菜狗 original author: Jacky Li Email : 3435673055qq.com Time of completion&#xff1a;2024.03.30 Last edited: 2024.03.30 目录 怎样完成票据证件的关键信息抽取任务 基于深度学习的主流方法 关键信息抽取任务流程 训练OCR模型…

用 AI 编程-释放ChatGPT的力量

最近读了本书&#xff0c;是 Sean A Williams 写的&#xff0c;感觉上还是相当不错的。一本薄薄的英文书&#xff0c;还真是写的相当好。如果你想看&#xff0c;还找不到&#xff0c;可以考虑私信我吧。 ChatGPT for Coders Unlock the Power of AI with ChatGPT: A Comprehens…

软考104-上午题-【结构化开发】-系统结构设计原则

一、系统结构设计原则 为保证总体结构设计顺利完成&#xff0c;应遵循以下几条原则。 1、分解-协调原则。整个系统是一个整体&#xff0c;具有整体目的和功能&#xff0c;但这些目的和功能的实现又是由相互联系的各个组成部分共同工作的结果。 解决复杂问题的一个很重要的原…

你该选择哪个职业呢?数据科学家、数据分析师和数据工程师

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

粉丝免费福利第一期-海浪型手机支架

&#x1f341; 作者&#xff1a;知识浅谈&#xff0c;CSDN签约讲师&#xff0c;CSDN博客专家&#xff0c;华为云云享专家&#xff0c;阿里云专家博主 &#x1f4cc; 擅长领域&#xff1a;全栈工程师&#xff0c;大模型&#xff0c;爬虫、ACM算法 &#x1f492; 公众号&#xff…

配置Web运行环境与第一个网页

目录 安装与配置Web环境: 1.下载 2.安装 3.下载插件 第一个网页: 安装与配置Web环境: 如下使用了VSC作为web的运行环境。 下面是VSC的官网点击进入:Download Visual Studio Code - Mac, Linux, Windowshttps://code.visualstudio.com/download 1.下载 进入官网后可以看到…

SpringBoot整合腾讯云邮件发送服务非STMP

SpringBoot整合腾讯云邮箱服务 1、pom配置 <!-- 腾讯云邮箱服务--><dependency><groupId>com.tencentcloudapi</groupId><artifactId>tencentcloud-sdk-java</artifactId><!-- go to https://search.maven.org/search?qtencen…

11-设计模式:Go常用设计模式概述

设计模式是啥呢&#xff1f;简单来说&#xff0c;就是将软件开发中需要重复性解决的编码场景&#xff0c;按最佳实践的方式抽象成一个模型&#xff0c;模型描述的解决方法就是设计模式。使用设计模式&#xff0c;可以使代码更易于理解&#xff0c;保证代码的重用性和可靠性。 …

动态规划之方格取数

方格取数 动态规划&#xff0c;数字三角形模型 题目链接 https://www.luogu.com.cn/problem/P1004 题目描述 解法一 O ( n 4 ) O(n^4) O(n4) #include<bits/stdc.h> using namespace std; int n, i, j, l, k, x, y, s; int d[55][55], f[55][55][55][55]; int main()…

nginx界面管理工具之nginxWebUI 搭建与使用

nginx界面管理工具之nginxWebUI 搭建与使用 一、nginxWebUI 1.nginx网页配置工具 官网地址: http://www.nginxwebui.cn 源码地址&#xff1a;https://git.chihiro.org.cn/chihiro/nginxWebUI 2.功能说明 本项目可以使用WebUI配置nginx的各项功能, 包括http协议转发, tcp协议…

CQ 社区版2.10.0 | 新增 SQL 审核、全新英文版上线…

三月中旬&#xff0c;我们预告了 CloudQuery 社区版即将上线的「SQL 审核」功能。现在&#xff0c;它来了&#xff01; 本次社区版 v2.10.0&#xff0c;除了 SQL 审核功能&#xff0c;我们还在手动授权、连接分组等模块做了新功能和优化。 新增功能 新增 SQL 审核功能 支持…

Linux编译器-gcc/g++/gdb使用

Linux编译器-gcc/g/gdb使用 一、背景知识二、 gcc如何完成2.1 预处理(进行宏替换)2.2 编译&#xff08;生成汇编&#xff09;2.3 汇编&#xff08;生成机器可识别代码&#xff09;2.4 连接&#xff08;生成可执行文件或库文件&#xff09; 三、函数库四、gcc选项五、gdb5.1 背景…

产品之美10| 小小提示词(hint),便于用户交互

最近AIGC功能火热&#xff0c;有文生图和图生图两种。当用户初次接触到文生图的时候&#xff0c;会有一刻停顿&#xff1a;我该怎用输入呢&#xff1f;这时候的hint就可以发挥作用了&#xff1a; 编辑框&#xff08;EditView)里面有可爱的小女孩&#xff0c;加风格卡通。用户看…

基于单片机的二维码LCD显示控制设计

**单片机设计介绍&#xff0c;基于单片机的二维码LCD显示控制设计 文章目录 一 概要二、功能设计设计思路 三、 软件设计原理图 五、 程序六、 文章目录 一 概要 基于单片机的二维码LCD显示控制设计是一个集硬件、软件与通信于一体的综合性项目。此设计的主要目标是实现单片机…

【动手学深度学习-pytorch】 9.4 双向循环神经网络

在序列学习中&#xff0c;我们以往假设的目标是&#xff1a; 在给定观测的情况下 &#xff08;例如&#xff0c;在时间序列的上下文中或在语言模型的上下文中&#xff09;&#xff0c; 对下一个输出进行建模。 虽然这是一个典型情景&#xff0c;但不是唯一的。 还可能发生什么其…