重要的字符(串)函数的使用及其实现

news2025/1/9 9:47:31

目录

字符串函数注意点

1、\0

2、适当使用const修饰

3、多使用assert断言

4、库函数不可能完全安全

1、求字符串长度strlen

 1、计数实现

2、递归实现

3、指针相减求元素个数

2、长度不受限制的字符串函数

1、strcpy

2、strcat

3、strcmp

3、长度受限制的字符串函数(字符函数)

1、strncpy

2、strncat

 3、strncmp

字符串函数注意点

1、\0

对于字符串来说,重要的是结束标志\0,在实现字符串函数时,要时刻考虑\0的影响。

是否要用\0标记某个位置,或者是否需要在最后补充一个\0。

2、适当使用const修饰

当我们给函数传入数组名时,若我们并不想修改这个数组的内容,可以在接收的指针前加上const来修饰,即保护我们的数组内容不被错误修改。

3、多使用assert断言

当代码量较大时,作为程序员我们是有可能传入一个空指针给函数的,如果我们在函数中对这个空指针解引用,非法访问一些空间,会使程序产生错误,但编译器往往无法精准定位到这个错误。    因此,出于我们自身考虑,多使用assert可减轻我们回头检查bug的负担,是一个编程好习惯。

4、库函数不可能完全安全

当我们完成一个库函数时,总能想到一些特殊的例子,导致库函数产生不安全的影响。也就是说,当一个喝醉酒的程序员写bug时,我们是拦不住的。

1、求字符串长度strlen

 1、计数实现

2、递归实现

3、指针相减求元素个数

刚学指针时我们知道,当两个指针指向相同的一块空间,且指针类型相同时,指针相减的绝对值是两个指针之间元素的个数。

总结:

1、求字符串长度时不用修改数组内容,或者当这个字符串为常量字符串时,它本身不能被修改,所以可用const来修饰。

2、求字符串长度len本身是一个非负的数,所以库函数中使用的是size_t返回类型,但是我们如果返回int也可以实现。同时,由于size_t的结果不能为负数,在进行进一步运算时,可能会产生错误

int main()
{
const char*str1 = "abcdef";
const char*str2 = "bbb";
if(strlen(str2)-strlen(str1)>0)
{
printf("str2>str1\n");
}
else
{
printf("srt1>str2\n");
}
return 0;

3、字符串函数的实现是到\0停止计数,且\0不算在内。注意与sizeof区别,sizeof是操作符,可通过类型返回一个确定的值。而strlen若找不到\0,就会一直找,导致产生随机值或error。

2、长度不受限制的字符串函数

1、strcpy

注意:不能与source在内存中重叠指的是,自己拷贝自己实际上没有意义。

 为保证目的地接收拷贝后仍为一个字符串,我们在涉及函数时需要拷贝完src的内容后补加\0

同时arr1所占空间大小必须大于arr2

 *dest++=*src++表达式的值为*src,当*src为\0时停止,即把\0补到拷贝字符串后面了。

注意:这里是后置++,赋值完\0后,src指向的是\0下一个字符,但我们不去访问,并不影响函数实现。 

2、strcat

注意:dest找到\0后覆盖,将src的\0也追加过去。

dest和src不能重叠,这是因为找到dest的\0覆盖后,src就失去了结束标志\0,导致追加过程死循环

这里必须保证arr1空间内容足够大,能够接收arr2的追加。

这里我们注意到,\0也会被追加过去。 

3、strcmp

 

注:比较的是第一个不同的字符的ASCLL码值的大小,不是字符串的长度。

只要相等,就继续比,有一个是\0即俩都是,return0,。不相等跳出来返回相应值。 

总结:长度不受限制的字符串是整体操作两个字符串,是有可能因为数组内存空间不够用,而造成非法访问的,从某种意义上来说是不够安全的。因此产生了下面长度受限制的字符串函数。即多引入一个参数n,控制src中操作字符串的长度,使其安全性提高。

3、长度受限制的字符串函数(字符函数)

1、strncpy

将src的前n个拷贝到dest,若n比src长,拷贝后继续补\0,若src比dest会导致溢出,也不是绝对安全。

  拷贝数超过dest溢出报错。

用int类型接收也可以完成功能。

2、strncat

 追加完要再补一个\0。若src长度不够,就全追加过去即可。

 

 3、strncmp

相比strcmp,多了一个与n个字符匹配就相同条件。

int my_strncmp(const char* p1, const char* p2, int k)
{
	assert(p1 && p2);
	while (*p1 == *p2 && k)
	{
		--k;
		if (*p1 == '\0'|| k==0)
		{
			return 0;
		}
			p1++;
			p2++;
	}
	return *p1 - *p2;
}
int main()
{
	char arr1[] = "abcdefg";
	char arr2[] = "abcdeeeeee";
	int ret =my_strncmp(arr1, arr2, 20);
	if (ret > 0)
	{
		printf("arr1>arr2\n");
	}
	else if (ret < 0)
	{
		printf("arr1<arr2\n");
	}
	else
	{
		printf("arr1=arr2\n");
	}
}

 实现时注意加上n这个条件

错误改正:

过两天再看一下自己的代码,发现strncpy有一个错误,当时调试时只用了k>len的例子。

当k<=len时,拷贝完字符,最后还要加一个\0

下面是修改过后的代码。

char* my_strncat(char* dest, const char* src, size_t k)
{
	assert(dest && src);
	char* start = dest;
	size_t len = strlen(src);
	while (*dest)
	{
		dest++;
	}
	if (k > len)
	{
		while (*dest++ = *src++)
		{
			;
		}
	}
	else
	{
		while (k)
		{
			--k;
			*dest++ = *src++;
		}
		*dest = '\0';
	}
	return start;
}
int main()
{
	char arr1[20] = "hello \0xxxxxxxx";
	char arr2[] = "world";
	my_strncat(arr1, arr2, 3);
}

多加上了一个*dest='\0'

如有其它错误,请读者尽快联系我加以改正。感谢大家的支持。

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

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

相关文章

ESP32设备驱动-MLX90614红外测温传感器驱动

MLX90614红外测温传感器驱动 1、MLX90614介绍 MLX90614 是一款用于非接触式温度测量的红外温度计。IR 敏感型热电堆检测器芯片和信号调节 ASIC 都集成在同一 TO-39 罐封装中。MLX90614 集成有低噪声放大器、17 位 ADC 和强大的 DSP 单元,因此温度计兼具高精度和高分辨率。 …

少儿Python每日一题(22):杨辉三角

原题解答 本次的题目如下所示: 杨辉三角形又称Pascal三角形,它的第i+1i+1行是的展开式的系数。 它的一个重要性质是:三角形中的每个数字等于它两肩上的数字相加。 下面给出了杨辉三角形的前4行: 1 1 1 1 2 1 1 3 3 1 给出n,输出它的前n行。 输入: 输入包含一个数n。 输出…

学习CSS3,使用双旋转实现福到了的迎春喜庆特效

春节快到了&#xff0c;因为疫情已经好久没有回老家了&#xff0c;今年终于可以回家过年了&#xff0c;我已经抑制不住自己激动的心情了。因此&#xff0c;我利用css3的旋转做了一个福到了的特效&#xff0c;而且是双旋转哦。 目录 1、实现思路 2、大红纸的渲染过程 3、错误…

【自学Python】Python运算符优先级

Python运算符优先级 Python运算符优先级教程 我们知道&#xff0c;在数学运算中&#xff0c;有 “先乘除后加减” 的运算规则&#xff0c;在我们程序语言中一样有运算符的优先级问题&#xff0c;来决定我们运算的顺序问题&#xff0c;这就是运算符的优先级。 即所谓运算符的…

C++STL之set与map的使用

本文目录前言一、关联式容器二、键值对(pair)三、树形结构的关联式容器1.set1.1set的介绍1.2set的使用1.2.1set的模板参数列表1.2.2set的构造(1)构造空的set(2)以数组的形式构造(3)用迭代器区间进行构造(4)拷贝构造1.2.3set的迭代器(1)iterator begin()(2)iterator end()(3)con…

一天变现100w就是这么简单!呆头鹅批量剪辑软件批量剪辑带货视频

呆头鹅剪辑是一款全自动的视频剪辑软件&#xff0c;包含剪辑、合成、去重、特效、配音、字幕、水印、后期处理、自动生成片头等功能&#xff0c;可以用于视频批量搬运&#xff0c;给视频增加特效&#xff0c;图片合成视频&#xff0c;视频混剪&#xff0c;自动加配音字幕&#…

Linux一看就会——make/Makefile

Linux一看就会——make/Makefile 背景 1.会不会写makefile&#xff0c;从一个侧面说明了一个人是否具备完成大型工程的能力。 2.一个工程中的源文件不计数&#xff0c;其按类型、功能、模块分别放在若干个目录中&#xff0c;makefile定义了一系列的 规则来指定&#xff0c;哪…

分享126个ASP源码,总有一款适合您

ASP源码 分享126个ASP源码&#xff0c;总有一款适合您 126个ASP源码DownLoad链接&#xff1a;https://pan.baidu.com/s/1wekzBbNE6JSFWtyLb_CdQg?pwdu1e0 提取码&#xff1a;u1e0 下面是文件的名字&#xff0c;我放了一些图片&#xff0c;文章里不是所有的图主要是放不下..…

I16-python中的Counter类

在很多场景中经常会用到统计计数的需求&#xff0c;比如在实现 KNN算法时统计 k 个标签值的个数&#xff0c;进而找出标签个数最多的标签值作为最终 kNN 算法的预测结果。Python内建的 collections 集合模块中的 Counter 类能够简洁、高效的实现统计计数。Counter 是 dict 字典…

Unity 3D 三维模型简介||

Unity 3D 三维模型简介 三维模型是用三维建模软件建造的立体模型&#xff0c;也是构成 Unity 3D 场景的基础元素。 Unity 3D 几乎支持所有主流格式的三维模型&#xff0c;如 FBX 文件和 OBJ 文件等。 开发者可以将三维建模软件导出的模型文件添加到项目资源文件夹中&#xf…

【服务器数据恢复】意外断电导致linux服务器故障的数据恢复案例

服务器数据恢复环境&故障&#xff1a; 某品牌730服务器&#xff0c;linux操作系统。 机房意外断电导致服务器部分文件丢失。 服务器数据备份&故障分析&#xff1a; 1、将linux服务器连接到准备好的数据恢复服务器上&#xff0c;以只读模式对服务器数据做镜像备份&#…

geoserver 图层访问权限控制

目录 一、环境 二、问题 三、设置登录才能访问图层 ①、创建身份验证过滤器 ②、创建过滤器 ③、校验 ④、过滤后的问题 一、环境 geoserver2.17.0 二、问题 1、geoserver图层不需要登录就能访问&#xff0c;是否存在安全隐患&#xff1f; 答&#xff1a;会有一定的安全隐…

【Linux操作系统】计算机体系结构和操作系统与进程概念深入理解

文章目录一.现代计算机体系结构1.和冯诺依曼体系结构的异同2.计算机的五大核心部件3.举例子:“我爱你”4.CPU,内存,磁盘的联系a.三者读写速度对比b.规定:CPU不直接和外设打交道二.操作系统1.操作系统三段论2.系统调用接口3.用户操作接口三.进程1.什么是进程?2.PCB3.查看进程4.…

222.完全二叉树的节点个数 |递归优化思路 + 复杂度分析

完全二叉树的节点个数 leetcode : https://leetcode.cn/problems/count-complete-tree-nodes/ 递归思路 递归的思路很简单, 假设们要统计一棵树的节点数, 那么 只要统计根节点的左子树的节点数, 和右子树的节点数加上根节点即可 那么, 假设我们要统计左子树的节点数, 其实就…

Centos7 Minimal 版本基本配置记录

每次搭测试环境之前都需要先装一台干净的虚拟机&#xff0c;然而 Centos7 Minimal 版本快速装完之后还需要配置&#xff1a;网络、国内源、一些基础工具&#xff08;net-tools、vim&#xff09;等才能远程连接和使用。记录一下&#xff0c;方便下次快速配置使用。 目录 1、网…

Docker镜像加载原理

文章目录什么是镜像 &#xff1f;Docker镜像加载原理UnionFS &#xff08;联合文件系统&#xff09;Docker镜像加载原理分层理解镜像Commit什么是镜像 &#xff1f; 镜像是一种轻量级、可执行的独立软件包&#xff0c;用来打包软件运行环境和基于运行环境开发的软件&#xff0…

C语言-自定义类型-枚举和联合(11.3)

目录 思维导图&#xff1a; 1.枚举 1.1 枚举类型的定义 1.2 枚举的优点 1.3 枚举的使用 2. 联合&#xff08;共用体&#xff09; 2.1 联合类型的定义 2.2 联合的特点 2.3 联合大小的计算 写在最后&#xff1a; 思维导图&#xff1a; 1.枚举 1.1 枚举类型的定义 例&…

Spring复习(三)

AOP AOP(Aspect Oriented Programming)面向切面编程&#xff0c;aop是一种设计思想&#xff0c;是oop面向对象编程的一种补充和完善&#xff0c;它通过预编译方式和运行期间动态代理的方式达成在不修改源代码的情况下&#xff0c;实现对业务逻辑的增强。 相关术语 横切关注点…

论文投稿指南——中文核心期刊推荐(农业基础科学)

【前言】 &#x1f680; 想发论文怎么办&#xff1f;手把手教你论文如何投稿&#xff01;那么&#xff0c;首先要搞懂投稿目标——论文期刊 &#x1f384; 在期刊论文的分布中&#xff0c;存在一种普遍现象&#xff1a;即对于某一特定的学科或专业来说&#xff0c;少数期刊所含…

【计组】存储器层次结构全景和局部性原理--《深入浅出计算机组成原理》(八)

课程链接&#xff1a;深入浅出计算机组成原理_组成原理_计算机基础-极客时间 目录 一、存储器层次结构全景 &#xff08;一&#xff09;SRAM &#xff08;二&#xff09;DRAM &#xff08;三&#xff09;存储器的层级结构 二、局部性原理 一、存储器层次结构全景 &…