详解C语言string.h中常见的13个库函数(上)

news2024/9/20 0:56:44

我计划讲解C语言string.h这个头文件中,最常见的13个库函数。为了让大家更加深入的理解这些函数,部分函数我会模拟实现。篇幅所限,如果文章太长了,可能会较难坚持读完,所以我会分几篇博客来讲述。本篇博客主要讲解的函数有:strlen, strcpy, strcat, strcmp。这四个函数是最基础,最常见的字符串操作函数,在字符串相关的场景中会频繁使用,希望大家在阅读完本篇博客后,对它们的原理、使用方式了如指掌,运用自如。考虑到有朋友可能不太理解C语言字符串的一些基础概念,本篇博客会先铺垫一些基础知识,已经熟悉C语言字符串的朋友们可以跳过。

字符串必备知识

什么是字符串?就是一串字符。C语言如何表示字符串呢?用双引号括起来一串字符即可。如:"hello world"就是一个字符串。

C语言中的字符串的结束标志是\0这个转义字符。比如"hello world"这个字符串,本质上,在内存中存储的是hello world\0,在所有的字符后面,会隐藏一个\0。注意:\0作为一个整体,是一个字符,它的ASCII码值是0。

字符串一般用字符数组来存储,比如:

char arr[] = "hello world";

就把"hello world"这个字符串存储到了字符数组arr里。此时数组arr的长度是多少呢?要把hello world这11个字符算上,后面还有一个\0,总共12个字符。但是,如果计算的是字符串的长度,是不算最后的\0的,长度就是11。

字符串的长度计算的是\0之前出现了几个字符,但是不包含\0本身!那如何求字符串长度呢?这就要引出今天要讲解的第一个库函数了。

string.h中的库函数

strlen

size_t strlen ( const char * str );

strlen是用来求字符串的长度的。只需要给它传字符串首字符的地址,它就会计算出该字符串的长度。注意:字符串常量,即单引号引起来的字符串,作为一个表达式,其值为首字符的地址,可以作为strlen的参数。除此之外,字符串也可以保存到字符数组中,数组名表示首元素地址,也可以作为参数。

字符串直接作为参数:

int len1 = strlen("abc"); // 3

char* str = "defg";
int len2 = strlen(str); // 4

数组名作为参数:

char arr[] = "abc";
int len = strlen(arr); // 3

strlen的返回值是size_t类型的。size_t是一个无符号整型。所以以下程序会输出什么?

if (strlen("abc") - strlen("abcd") < 0)
	printf("<\n");
else
	printf(">=");

看起来应该是“小于”,但结果是“大于等于”,原因是无符号整型的差还是无符号整型,一定是大于等于0的。

下面我们来模拟实现strlen。实现思路很简单,从首字符开始,向后数,直到遇到\0就停下来。

size_t my_strlen(const char* str)
{
	assert(str);
	
	size_t count = 0;
	while (*str)
	{
		++count;
		++str;
	}
	
	return count;
}

当然,我们也可以一直向后找\0,根据“指针-指针得到的是指针之间的元素个数”的原理,用\0的地址减首字符的地址,也可以得到字符串的长度。

size_t my_strlen(const char* str)
{
	assert(str);
	
	const char* eos = str; // end of str
	while (*eos)
	{
		++eos;
	}
	
	return eos - str;
}

当然,如果不创建临时变量,也可以使用递归实现,这种实现并不推荐,因为递归是有缺陷的,递归深度太深可能导致栈溢出。递归实现思路是:字符串的长度=1+从下一个字符开始数的长度。也就是说,strlen(str) = 1+strlen(str+1)。当然,如果str是空字符串,即*str=='\0',长度就为0。

size_t my_strlen(const char* str)
{
	assert(str);
	
	if (*str)
		return 1 + my_strlen(str + 1);
	else
		return 0;
}

strlen总结:

  1. strlen是用来求字符串长度的,会返回\0之前出现了几个字符。
  2. 参数指向的字符串必须以\0结尾,否则结果是随机值。
  3. 函数的返回值类型是size_t的,size_t是一个无符号整型。

strcpy

char * strcpy ( char * destination, const char * source );

strcpy是用来完成字符串拷贝的。它有2个参数,分别是目的地和起始位置。比如,把字符串arr1拷贝到arr2里,要这么写:

char arr1[20] = {0};
char arr2[] = "abc";
strcpy(arr1, arr2);

注意:拷贝时,会把arr2中的"abc"拷贝到arr1中,包括结尾的\0

strcpy会返回目标空间的起始地址,方便函数的链式访问,比如:

printf("%s\n", strcpy(arr1, arr2));

以上代码,在把arr2中的字符串拷贝到arr1中后,顺便把arr1打印出来,看看有没有拷贝成功。

下面讲讲模拟实现。其实重点是拷贝的过程,也就是*dst++ = *src++,把src指向的字符拷贝到dst指向的空间中,并且2个指针向后走,直到src遇到\0,此时结束循环,返回起始地址。注意dst在拷贝的过程中一直在向后走,所以需要在最开始先保存下来,方便最后返回目标空间的地址。

char* my_strcpy(char* dst, const char* src)
{
	assert(dst && src);

	char* ret = dst;
	while (*dst++ = *src++)
	{
		;
	}
	return ret;
}

strcpy总结:

  1. strcpy是用来拷贝字符串的。
  2. 源字符串必须以\0结束,否则会一直拷贝字符,直到遇到内存中的\0
  3. 拷贝时,会把源字符串结尾的\0也拷贝到目标空间中。
  4. 目标空间必须足够大,能够容纳拷贝的源字符串,否则会导致内存的越界访问。
  5. 目标空间必须可变(可修改)。

strcat

char * strcat ( char * destination, const char * source );

strcat是完成字符串追加的,它可以在目标字符串后面追加源字符串。可以理解为,先从目标空间中找到\0,即目标字符串的结尾,然后从\0所在的位置开始,向后追加源字符串。比如:在abc后面追加def后就得到了abcdef,追加的时候会把源字符串的\0也追加过去。函数会返回目标字符串的起始地址。

其实可以简单理解为:先找到目标字符串结尾的\0,然后从把源字符串以strcpy的方式拷贝到目标字符串后面,大家看到模拟实现后就明白了。

char* my_strcat(char* dst, const char* src)
{
	assert(dst && src);
	char* ret = dst;

	while (*dst)
	{
		++dst;
	}

	while (*dst++ = *src++)
	{
		;
	}
	return ret;
}

其实就是在strcpy的模拟实现的基础上,加上了下面的代码,即找dst中的\0

while (*dst)
{
	++dst;
}

根据以上的实现,能不能自己给自己追加呢?比如:

char arr[10] = "abc";
strcat(arr, arr);

我们想再arr后面追加arr,预期结果是,“abc"后面追加"abc"得到"abcabc”,但是根据以上模拟实现的代码,在追加的同时,会把src中的\0给覆盖掉,所以源字符串中就内有\0了,在拷贝的时候,本来是遇到\0就停止了,但是一直找不到\0,就导致无限循环。

那如何实现自己给自己追加呢?这就要用到strncpy函数了,这个函数我会在下一篇博客中介绍。

strcat总结:

  1. strcat是用来追加字符串的,会把源字符串追加到目标字符串后面。
  2. 源字符串和目标字符串都必须以\0结束。
  3. 目标空间必须足够大,以容纳追加的字符串。
  4. 目标空间必须可变(可修改)。
  5. 不能自己给自己追加。

strcmp

int strcmp ( const char * str1, const char * str2 );

strcmp是用来比较2个字符串的。比较方式是:从第一个字符开始,一个一个往后比,直到遇到第一对不同的字符或者都遇到\0。如果遇到第一对不同的字符,则比较其ASCII码值,哪个大,对应的字符串就更大;如果都遇到\0,则2个字符串相等。函数会根据不同的大小关系返回不同的值,当str1>str2时,返回一个正整数;如果str1<str2,返回一个负整数;如果str1==str2,返回0。

根据以上的描述,可以模拟实现,如下:

int my_strcmp(const char* str1, const char* str2)
{
	assert(str1 && str2);

	while (*str1 == *str2)
	{
		if (*str1 == '\0')
			return 0;

		++str1;
		++str2;
	}

	return *str1 - *str2;
}

strcmp总结:

  1. strcmp是用来比较2个字符串的大小的。
  2. 假设从左到右2个参数分别是str1和str2。如果str1>str2,返回正整数;如果str1<str2,返回负整数;如果str1==str2,返回0。
  3. 比较时,是根据从2个字符串的第一个字符开始比较,向后比,直到遇到不同的字符或者都遇到\0。如果遇到不同的字符,则ASCII码值大的字符对应的字符串更大;如果都遇到\0,则2个字符串相等。
  4. 2个字符串结尾都应该有\0,没有的话,会一直向后比较字符,如果都相等,会一直比下去。

总结

  1. strlen是用来求字符串长度的,统计的是\0之前出现的字符个数。
  2. strcpy是用来拷贝字符串的。有2个参数,目标空间在前面,源字符串在后面;strcat是用来追加字符串的,也是2个参数,目标字符串在前面,源字符串在后面,比strcpy多了1个步骤,即先在目标空间中找到\0,再拷贝。
  3. strcmp是用来比较字符串的。前面字符串和后面字符串比较,如果前面大会返回正整数,前面小会返回负整数,前后相等返回0。

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

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

相关文章

什么是全民开发?|概念、技能和优势

注&#xff1a;全民开发的英文是Citizen Development&#xff0c;由咨询公司Gartner在2010年提出的概念&#xff0c;指非专业开发人员使用低代码或无代码平台创建应用程序&#xff0c;无需IT部门的支持&#xff0c;旨在提高生产力并降低开发成本。 国内普遍将Citizen Developme…

【ROS】单目摄像机的标定

在上节: ROS 教程之 vision : 用笔记本摄像头获取图像 能够使用相机后&#xff0c;就需要对相机进行标定&#xff0c;在ROS上使用原始图像校准单目相机。 操作步骤 1、首先将catkin_ws/src/usb_cam/launch/usb_cam-test.launch的文件内容修改掉。 原有内容&#xff1a; <…

docker 使用Dockerfile 部署springboot项目

1、先准备好你的springboot项目jar包。拉取 openjdk docker pull openjdk:8-jdk-alpine 2、上传你的springboot项目&#xff0c;然后配置 Dockerfile&#xff1a; FROM openjdk:8-jdk-alpine ADD ./springbootstudy.jar.jar /app.jar ENTRYPOINT ["java","-jar…

【源码】Spring Cloud Gateway 是在哪里匹配路由的?

我们知道&#xff0c;经过网关的业务请求会被路由到后端真实的业务服务上去&#xff0c;假如我们使用的是Spring Cloud Gateway&#xff0c;那么你知道Spring Cloud Gateway是在哪一步去匹配路由的吗&#xff1f; 源码之下无秘密&#xff0c;让我们一起从源码中寻找答案。 入…

Ant Design 常见用法与坑点总结(二):Form 表单下拉框设置初始值

前言 Ant Design 是蚂蚁出品的出色优秀的 React 组件库&#xff0c;相信使用 React 进行管理系统开发的小伙伴们或多或少都接触过 Ant Design。很多公司基于 React 开发的管理端系统也都是使用 Ant Design 的组件库。 因此&#xff0c;了解 Ant Design 的常见用法与坑点还是有…

react中如何系统化的处理时间操作?

在 Web 开发中&#xff0c;我们经常需要处理日期和时间的格式化。 在 React 中&#xff0c;这个过程变得更加容易和直观&#xff0c;因为我们可以使用一个叫做 moment 的 npm 包来帮助我们完成这个任务。 什么是 Moment? Moment.js是一个JavaScript库&#xff0c;用于处理日…

使用FFMPEG库将PCM编码为AAC

准备 ffmpeg 版本4.4 准备一段48000Hz 2 channel f32le 格式的PCM原始数据 这里我们直接使用ffmpeg命令行提取 ffmpeg -i beautlWorld.mp4 -ar 48000 -ac 2 -f f32le 48000_2_f32le.pcm -ac 采样率 -ac 音频通道 -f f32le 音频样本数据存储格式&#xff08;f32 ---- float…

【OJ比赛日历】快周末了,不来一场比赛吗? #04.22-04.28 #11场

CompHub 实时聚合多平台的数据类(Kaggle、天池…)和OJ类(Leetcode、牛客…&#xff09;比赛。本账号同时会推送最新的比赛消息&#xff0c;欢迎关注&#xff01; 更多比赛信息见 CompHub主页 或 点击文末阅读原文 以下信息仅供参考&#xff0c;以比赛官网为准 目录 2023-04-22&…

程序员最新赚钱指南!

程序员们的主要收入来源 1️⃣首先&#xff0c;我们要明白程序员无论编程开发多么努力&#xff0c;随着时间推移&#xff0c;受年龄、生活、健康等因素&#xff0c;程序员们都会面临职业天花板&#xff0c;这是大多数人不可规避的一个事实。 2️⃣其次&#xff0c;这几年因为…

【Python】【进阶篇】二十三、Python爬虫的Selenium库

目录 二十三、Python爬虫的Selenium库23.1 Selenium下载安装23.2 安装浏览器驱动23.3 自动访问百度 二十三、Python爬虫的Selenium库 Selenium 是一个用于测试 Web 应用程序的自动化测试工具&#xff0c;它直接运行在浏览器中&#xff0c;实现了对浏览器的自动化&#xff0c;它…

集群和分布式

本文以即时通讯软件&#xff08;IM&#xff09;为例&#xff0c;介绍单机、集群、分布式的区别&#xff0c;以及它们各自的优缺点。 假设现在开发一款IM&#xff0c;刚开始业务比较简单&#xff0c;用户量也较少&#xff0c;我们将服务部署在一台单机服务器上足矣。软件开发过程…

【从零开始学Skynet】实战篇《球球大作战》(十):agent代码设计

现在开发登录流程涉及的最后一个服务agent&#xff0c;完成后就可以真正地把框架运行起来了。还会演示agent的单机功能&#xff0c;做个“打工”小游戏。 1、消息分发 玩家登录后&#xff0c;gateway会将客户端协议转发给agent&#xff08;流程图的阶段⑨&#xff09;。 新建se…

第五章-数字水印-1-位平面

数字水印概念 数字水印是一种将特定数字信息嵌入到数字作品中从而实现信息隐藏、版权认证、完整性认证、数字签名等功能的技术。 以图片水印为例: 水印嵌入过程:版权信息水印A嵌入到图像B中,得到含有水印的图像C,图像C与图像B在外观上基本一致&#xff0c;肉眼无法区分差异。…

OpenHarmony的线程间通信EventHandler

一、初识EventHandler ​ 在OpenHarmony的开发过程中&#xff0c;如果遇到处理下载、运算等较为耗时的操作时&#xff0c;会阻塞当前线程&#xff0c;但是实际操作中又不希望当前线程受到阻塞。比如&#xff1a;我们的app在界面上有一个下载文件的处理按钮&#xff0c;如果在按…

计算机网络基础 第三章练习题

计算机网络基础 第三章练习题 现在大量的计算机是通过诸如以太网这样的局域网连入广域网的&#xff0c;而局域网与广城网的互联是通过( A)实现的。 A. 路由器B. 资源子网C. 桥接器D. 中继器 下列不属于数据链路层功能的是(B )。 A. 帧定界功能B. 电路管理功能C. 差错控制功能D…

HCIP——交换

交换 园区网架构 交换机实现了一下功能 无限的传输距离——识别&#xff0c;重写电信号&#xff08;帧&#xff09;保证信息完整彻底解决了冲突二层单播——MAC地址表提高端口密度 MAC 单播地址&#xff1a;MAC地址第一个字节第8位为0 组播地址&#xff1a;MAC地址第一个字…

Camera | 8.让rk3568支持前后置摄像头

一、目标 本文主要目标是&#xff0c;支持前置摄像头0v5648、后置摄像头ov13850&#xff0c;以及移植过程遇到的一些小问题的解决。 1. 摄像头连接图 参考上图&#xff0c;摄像头详细信息如下&#xff1a; 2个摄像头均连接在I2C通道42个摄像头共用同一个MIPI数据通道2个摄像…

C++——探究引用

文章目录 概述引用的概念引用特性引用的作用**引用做参数****引用作为函数返回值** 常引用引用的底层实现总结一下引用和指针的不同点 概述 本篇博客将讲述c相对于c新增的一个重要的内容——引用&#xff0c;深入研究其语法细节以及其需要注意的一些要点。 引用的概念 竟然要学…

Baumer工业相机堡盟工业相机如何联合BGAPISDK和佳能EF变焦镜头实现相机的自动变焦(C++)

Baumer工业相机堡盟工业相机如何联合BGAPISDK和佳能EF变焦镜头实现相机的自动变焦&#xff08;C&#xff09; Baumer工业相机Baumer工业相机BGAPISDK中控制变焦镜头的技术背景代码案例分享第一步&#xff1a;开启相机自动调焦功能模块第二步&#xff1a;控制自动变焦镜头电机的…