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

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

时间:2024-03-28 18:34:05浏览次数:26  
标签:02 字符 极练 end offer 空格 char 字符串 替换

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

                                                               文章所属专栏:《剑指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 here
    if(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 here
    int 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是字符串的长度。因此,这个算法是高效且实用的。

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

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

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

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

三. 完结散花

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

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

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

我们下期不见不散~~

标签:02,字符,极练,end,offer,空格,char,字符串,替换
From: https://blog.csdn.net/2301_80221228/article/details/137016372

相关文章

  • 16,2024年Python大厂面试分享
    6.3.路由6.3.1.配置分布式路由在tedu_note/urls.py中,将所有user/***相关路由转交给user处理fromdjango.contribimportadminfromdjango.urlsimportpath,includeurlpatterns=[path(‘admin/’,admin.site.urls),path(‘user/’,include(‘user.urls’))......
  • 20240328每日一题题解
    20240328每日一题题解摘要本文对于2024年3月28日的每日一题进行了问题重述,并将问题拆解为五个步骤,分别进行了详细的讨论与求解,实现了整型与字符串类型的互相转换。并且还指出,在编写C++程序时,需要观察数据范围,在有必要时使用长整型(longlong)存储数据,以免出现整型溢出现象。关键......
  • 2024最新最全Java和Go面经,面试了30多场,终于上岸了!
    ​>本文来自我们技术交流群群友的投稿,未经授权,禁止转载。原文链接:太难了,Java和Go,面试了30多场,终于上岸了!先听一下TA的故事2023年10月份我就做好了离职跳槽的准备,做了3年Java后端开发的我,对自己的技术能力还是很有底气的。之前虽不是一线大厂,也算是比较知名的中厂了。加上前公......
  • 2024 VEXIQ 赛季笔(游)记 Pt.1
    2024/03/07老师让我们做机器初步思考了。搞搞戒指,只要一个小夹子加上赛季的抬升吸环改一下就可以了,方便的一批。于是夹子10分钟不到搞完了,现在是缝合怪时间。但是老师下课不让我搞了/kk。2024/03/14学校pi-day为什么不给pi吃?学校pi-day为什么不给pi吃?学校pi-day......
  • 【专题】2024年3月数字化行业报告合集汇总PDF分享(附原数据表)
    原文链接:https://tecdat.cn/?p=35531原文出处:拓端数据部落公众号在科技浪潮的推动下,人工智能行业正在经历着前所未有的变革与发展。从自然语言处理到数字社交,再到AI数字人、绿色智能制造等多个领域,人工智能正逐渐渗透到我们生活的各个角落。然而,这一过程中也伴随着新的挑战和问......
  • 【专题】2022年中国制造业数字化转型研究报告PDF合集分享(附原数据表)
    报告链接:http://tecdat.cn/?p=32145本文中所说的制造业数字化转型,指的是在制造企业的设计、生产、管理、销售及服务的每一个环节中,将新一代信息技术应用到制造企业的设计、生产、管理、销售及服务的每一个环节中,并可以以每一个环节中产生的数据为基础,展开控制、监测、检测、预测......
  • dubhe2024 BuggyAllocator:通过修改_IO_2_1_stdout_的内容进行任意读
    在堆题中遇到没有show()函数的情况,导致无法泄露地址。这时可以通过修改_IO_2_1_stdout_来强制程序输出一段内存,从而泄露需要的地址。例题:dubhe2024BuggyAllocatordubhe2024,xctf分站赛最后一场凄惨爆零,主看了这道题一整天,逆清楚了但找不到漏洞。事后来看当时就算找到洞了也不会......
  • [NOI2002] 荒岛野人
    [NOI2002]荒岛野人传送门题目描述克里特岛以野人群居而著称。岛上有排列成环行的\(m\)个山洞。这些山洞顺时针编号为\(1,2,\dots,m\)。岛上住着\(n\)个野人,一开始依次住在山洞\(C_1,C_2,\dots,C_n\)中,以后每年,第\(i\)个野人会沿顺时针向前走\(P_i\)个洞住下来。......
  • 2024.03.28【UI设计】新拟态风格设计
    新拟态风格就是类似于给图形图案制作出3D的凸出或者凹进效果的风格这个风格的设计需要使用到即时设计软件的蒙版(与ai不同,ai的蒙版仅有透明度蒙版,无轮廓蒙版)新拟态风格的实现主要是通过三个效果:(1)一个相对浅色高斯模糊效果元素、(2)一个相对深色的无效果元素、(3)一个正常颜色的高......
  • 最佳开源文档管理解决方案:2024年免费软件精选
    本文将为大家分享9款开源文档管理系统:Bitrix24、Kimios、OpenDocMan、Papermerge、Nuxeo、OpenKM、Teedy、FileRun、SeedDMS。在现今充满数字化的世界里,不论大小,各种组织都会产出很多文件、图片等数字化内容。好好管理这些信息对于组织的顺畅运作和保持其竞争力非常关键。这......