详解C语言10大字符串函数【超详细建议点赞收藏】

news2025/3/12 23:59:58

目录

  • 1. strlen----求字符串长度
    • 1.1 函数介绍
    • 1.2 函数使用
    • 1.3 模拟实现
  • 2. strcpy----字符串拷贝
    • 2.1 函数介绍
    • 2.2 函数使用
    • 3.3 模拟实现
  • 3. strcat----字符串追加
    • 3.1 函数介绍
    • 3.2 函数使用
    • 3.3 模拟实现
  • 4. strcmp----字符串比较
    • 4.1 函数介绍
    • 4.2 函数使用
  • 5. strncpy----长度受限的字符串拷贝
    • 5.1 函数介绍
    • 5.2 函数使用
  • 6. strncat----长度受限的字符串追加
    • 6.1 函数介绍
    • 6.2 函数使用
  • 7. strncmp----长度受限的字符串比较
    • 7.1 函数介绍
    • 7.2 函数使用
  • 8. strstr----子字符串查找
    • 8.1 函数介绍
    • 8.2 函数使用
    • 8.3 模拟实现
  • 9. strtok----字符串切割
    • 9.1 函数介绍
    • 9.2 函数使用
  • 10. strerror----错误信息报告
    • 10.1 函数介绍
    • 10.2 函数使用

1. strlen----求字符串长度

1.1 函数介绍

![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/7e2ae198be464842a21bba027a9b9fcc.png

  • 注意:要引头文件<string.h>
  • 字符串已经 ‘\0’ 作为结束标志,strlen函数返回的是在字符串中 ‘\0’ 前面出现的字符个数(不包含 ‘\0’ )。
  • 参数指向的字符串必须要以 ‘\0’ 结束。
  • 注意:函数的返回值为size_t,是无符号的( 易错

1.2 函数使用

正确使用:

#include <stdio.h>
#include <string.h>

int main()
{
	char arr[] = "abcdef";
	int len = strlen(arr);
	printf("%d\n", len);

	return 0;
}

错误使用1:

#include <stdio.h>
#include <string.h>

int main()
{
	char arr[] = {'a','b','c','d'};
	int len = strlen(arr);
	printf("%d\n", len);

	return 0;
}

注意:像这样初始化,arr数组里没有’\0’,在计算时不知道在哪里结束,输出结果会是随机值

错误使用2:

#include <stdio.h>
#include <string.h>

int main()
{
	char arr1[] = "abc";
	char arr2[] = "abcdef";
	 
	if (strlen(arr1) - strlen(arr2) > 0)
		printf(">\n");
	else 
		printf("<\n");

	return 0;
}

注意:这个代码的输出结果是>。这是因为strlen返回的是无符号整型,-3转换成无符号整型是一个非常大的数字,所以必定大于零。

1.3 模拟实现

方式1:

#include <stdio.h>

//使用计数器
size_t my_strlen(char* str)
{
	assert(str);
	size_t count = 0;

	while (*str != '\0')
	{
		count++;
		str++;
	}

	return count;
}

int main()
{
	char arr[] = "abcdef";
	size_t n = my_strlen(arr);
	printf("%zu\n", n);

	return 0;
}

方式2:

//使用递归
int my_strlen(const char * str)
{
 assert(str);
 if(*str == '\0')
 return 0;
 else
 return 1+my_strlen(str+1);
}

方式3:

//使用指针-指针的⽅式
int my_strlen(char *s)
{
 assert(str);
 char *p = s;
 while(*p != ‘\0)
 p++;
 return p-s;
}

2. strcpy----字符串拷贝

2.1 函数介绍

![![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/e5d8de9b63e74e768165de028a5f440c.png](https://img-blog.csdnimg.cn/direct/3d6a558927d34d029f1e4e0f9c2a0148.png

  • 注意:要引头文件<string.h>
  • 源字符串必须以 ‘\0’ 结束。
  • 会将源字符串中的 ‘\0’ 拷贝到目标空间
  • 目标空间必须足够大,以确保能存放源字符串。
  • 目标空间必须可变。
  • 返回值是指向目标字符串的第一个字符的地址。

2.2 函数使用

#include <stdio.h>
#include <string.h>

int main()
{
	char arr1[] = "abcdef";
	char arr2[30] = { 0 };
	strcpy(arr2, arr1);
	printf("%s\n", arr2);

	return 0;
}

3.3 模拟实现


char* my_strcpy(char* dest, char* src)
{
	assert(dest && src);
	char* ret = dest;
	while (*src)
	{
		*dest++ = *src++;
	}
	*dest = *src;
	return ret;
}

int main()
{
	char arr1[] = "abcdef";
	char arr2[40] = { 0 };

	char* p = my_strcpy(arr2, arr1);
	printf("%s\n", arr2);

	return 0;
}

3. strcat----字符串追加

3.1 函数介绍

在这里插入图片描述

  • 注意:要引头文件<string.h>
  • 源字符串必须以 ‘\0’ 结束。
  • 目标空间必须有足够的大,能容纳下源字符串的内容。
  • 目标空间必须可修改。
  • 返回值是指向目标字符串的第一个字符的地址。

3.2 函数使用

#include <stdio.h>
#include <string.h>

int main()
{
	char arr1[40] = "hello ";
	char arr2[] = "world";
	strcat(arr1, arr2);
	printf("%s\n", arr1);

	return 0;
}

思考:如果自己给自己追加,可以吗?

答案是:不行!因为自己给自己追加时会破坏源字符串的内容,源字符串的’\0’被覆盖了,会造成死循环。

3.3 模拟实现

char* my_strcat(char* dest, char* src)
{
	assert(dest && src);
	char* ret = dest;
	while (*dest!='\0')
	{
		dest++;
	}
	while (*src)
	{
		*dest++ = *src++;
	}
	*src = *dest;

	return ret;
}
int main()
{
	char arr1[30] = "hello ";
	char arr2[] = "world";

	char* p = my_strcat(arr1, arr2);
	printf("%s\n", arr1);

	return 0;
}

4. strcmp----字符串比较

4.1 函数介绍

在这里插入图片描述

  • 注意:要引头文件<string.h>
  • 标准规定:
    -第一个字符串大于第二个字符串,则返回大于0的数字;
  • 第一个字符串等于第二个字符串,则返回0;
  • 第一个字符串小于第二个字符串,则返回小于0的数字。
  • 返回值是有符号的整型。

4.2 函数使用

#include <stdio.h>
#include <string.h>

int main()
{
	char arr1 = "abcdef";
	char arr2 = "abcdeg";
	int ret = strcmp(arr1, arr2);
	if (ret > 0)
		printf(">\n");
	else if (ret < 0)
		printf("<\n");
	else
		printf("==\n");

	return 0
}

5. strncpy----长度受限的字符串拷贝

5.1 函数介绍

在这里插入图片描述

  • 注意:要引头文件<string.h>
  • 拷贝num个字符从源字符串到目标空间。
  • 如果源字符串的长度小于num,则拷贝完源字符串之后,在目标的后边追加’\0’,直到num个。
  • 返回值是指向目标字符串的第一个字符的地址。

5.2 函数使用

#include <stdio.h>
#include <string.h>

int main()
{
	char arr1[] = "abcdef";
	char arr2[] = "bit";

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

	return 0;
}

注意:如果传递的num大于要拷贝的字符个数,则拷贝完字符串后会自动添加’\0’。

6. strncat----长度受限的字符串追加

6.1 函数介绍

在这里插入图片描述

注意:

  • 注意:要引头文件<string.h>
  • 追加完之后末尾会重新加’\0’。
  • 如果传递的num大于源字符串的字符个数,则追加完字符串后不会自动添加’\0’。

6.2 函数使用

#include <stdio.h>
#include <string.h>

int main()
{
	char arr1[30] = "abcdef ";
	char arr2[] = "hello";

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

	return 0;
}

7. strncmp----长度受限的字符串比较

7.1 函数介绍

在这里插入图片描述

7.2 函数使用

#include <stdio.h>
#include <string.h>

int main()
{
	char arr1[] = "abcde";
	char arr2[] = "abcd";

	int ret = strncmp(arr1, arr2, 5);
	if (ret == 0)
		printf("==\n");
	else if (ret > 0)
		printf(">\n");
	else
		printf("<\n");

	return 0;
}

8. strstr----子字符串查找

8.1 函数介绍

在这里插入图片描述

  • 注意:要引头文件<string.h>
  • 功能是查找str1里有没有存在str2字符串(子串)。
  • 若存在,则会返回子串的第一个字符的地址,若不存在,则返回NULL。

8.2 函数使用

#include <stdio.h>
#include <string.h>

int main()
{
	char str1[] = "abbbcd";
	char str2[] = "bbc";
	char* p = strstr(str1, str2);
	printf("%s\n", p);

	return 0;
}

输出结果:
在这里插入图片描述

8.3 模拟实现

char* my_strstr(const char* str1, const char* str2)
{
	assert(str1 && str2);
	const char* s1 = str1;
	const char* s2 = str2;
	const char* p = str1;

	while (*p)
	{
		s1 = p;
		s2 = str2;
		while (*s1 != '\0' && *s2 != '\0' && *s1 == *s2)
		{
			s1++;
			s2++;
		}
		if (*s2 == '\0')
		{
			return (char*)p;
		}
		p++;
	}

	return NULL;
}

int main()
{
	char arr1[] = "abbcdef";
	char arr2[] = "bbc";

	char* p = my_strstr(arr1, arr2);
	if (p == NULL)
		printf("子串找不到\n");
	else
		printf("%s\n", p);

	return 0;
}

9. strtok----字符串切割

9.1 函数介绍

在这里插入图片描述

  • 注意:要引头文件<string.h>
  • 第二个参数是个字符串,定义了用作分隔符的字符集合
  • 第一个参数指定一个字符串,它包含了0个或者多个delimiters字符串中一个或者多个分隔符分割的标记;
  • strtok函数找到str中的下一个标记,并将其用 \0 结尾,返回一个指向这个标记的指针。(注:strtok函数会改变被操作的字符串,所以在使用strtok函数切分的字符串一般都是临时拷贝的内容并且可修改。)
  • strtok函数的第一个参数不为 NULL ,函数将找到str中第一个标记,strtok函数将保存它在字符串
    中的位置;
  • strtok函数的第一个参数为 NULL ,函数将在同一个字符串中被保存的位置开始,查找下一个标记;
  • 如果字符串中不存在更多的标记,则返回 NULL 指针。

9.2 函数使用

#include <stdio.h>
#include <string.h>

int main()
{
	char arr1[40] = "caochunfan@sge.bit";
	const char* cp = "@.";
	char tmp[40] = { 0 };
	strcpy(tmp, arr1);
    
    //下面注释的代码可优化为:
	for (char* ret = strtok(arr1, cp);
		ret != NULL;
		ret = strtok(NULL, cp))
	{
		printf("%s\n", ret);
	}
	//char* ret = strtok(tmp, cp);
	//if (ret != NULL)
	//	printf("%s\n", ret);

	// ret = strtok(NULL, cp);
	//if (ret != NULL)
	//	printf("%s\n", ret);

	// ret = strtok(NULL, cp);
	//if (ret != NULL)
	//	printf("%s\n", ret);

	return 0;
}

输出结果:
在这里插入图片描述

如果两个分隔符相邻,但是中间没有任何字符串,则会直接跳过:

#include <stdio.h>
#include <string.h>

int main()
{
	char arr1[40] = "caochunfan@@sge.bit";
	const char* cp = "@.";
	char tmp[40] = { 0 };
	strcpy(tmp, arr1);

	for (char* ret = strtok(arr1, cp);
		ret != NULL;
		ret = strtok(NULL, cp))
	{
		printf("%s\n", ret);
	}

	return 0;
}

输出结果:
在这里插入图片描述

10. strerror----错误信息报告

10.1 函数介绍

在这里插入图片描述

  • 注意:要引头文件<string.h>
  • c语言的库函数,在执行失败时,都会设置错误码,比如0,1,2,3…每个错误码对应不同的错误信息

10.2 函数使用

#include <stdio.h>
#include <string.h>

int main()
{
	printf("%s\n", strerror(0));
	printf("%s\n", strerror(1));
	printf("%s\n", strerror(2));
	printf("%s\n", strerror(3));
	printf("%s\n", strerror(4));

	return 0;
}

输出结果:

在这里插入图片描述

  • 注意:errno:C语言设置的一个全局的错误码存放变量。使用时要引用头文件<errno.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>

int main()
{
	//errno:c语言设置的一个全局的错误码存放变量
    //下面伪代码:
FILE* pf = fopen("test,txt", "r");
if (pf == NULL)
{
	printf("%s\n", strerror(errno));
	return 1;
}
else
{
	//
}

	return 0;
}

输出结果:
在这里插入图片描述
因为我没有在该路径下新建"test,txt"文件,所以会有错误信息报告"No such file or directory"。

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

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

相关文章

一、直方图相关学习

1、灰度直方图 1.1 基本概念和作用 表示图像中每个灰度级别的像素数量。用于分析图像的亮度分布情况。 1.2 代码示例 参数介绍 hist cv2.calcHist(images, channels, mask, histSize, ranges, hist, accumulate)-images&#xff1a;输入图像的列表。对于灰度图像&#xff0…

APP端网络测试与弱网模拟

当前APP网络环境比较复杂&#xff0c;网络制式有2G、3G、4G网络&#xff0c;还有越来越多的公共Wi-Fi。不同的网络环境和网络制式的差异&#xff0c;都会对用户使用app造成一定影响。另外&#xff0c;当前app使用场景多变&#xff0c;如进地铁、上公交、进电梯等&#xff0c;使…

基于Springboot的新能源充电系统(有报告)。Javaee项目,springboot项目。

演示视频&#xff1a; 基于Springboot的新能源充电系统&#xff08;有报告&#xff09;。Javaee项目&#xff0c;springboot项目。 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构&a…

AtCoder Regular Contest 172(仅A题)

A - Chocolate 给N个人分巧克力&#xff0c;分的大小是2^Ai * 2^Ai&#xff0c;给你一个大小为H*W的巧克力&#xff0c;问能不能给N个人都分到要求的巧克力。 假设蓝色是h*w的大巧克力&#xff0c;红色的是要分出来的巧克力&#xff08;四个角落都一样&#xff0c;这里用左上…

VMware虚拟机安装CentOS7

对于系统开发来说&#xff0c;开发者时常会需要涉及到不同的操作系统&#xff0c;比如Windows系统、Mac系统、Linux系统、Chrome OS系统、UNIX操作系统等。由于在同一台计算机上安装多个系统会占据我们大量的存储空间&#xff0c;所以虚拟机概念应运而生。本篇将介绍如何下载安…

innoDB page页结构详解

Page是整个InnoDB存储的最基本构件,也是InnoDB磁盘管理的最小单位,与数据库相关的所有内容都存储在这种Page结构里。 Page分为几种类型,常见的页类型有数据页(B+tree Node)Undo页(Undo Log Page)系统页(System Page) 事务数据页(Transaction System Page)等 Page 各…

重学Java 15.面向对象.3.数组常用算法

我走过漫漫求学路 大雪落在求知所依靠的心脏上 昨日种种我弃之如敝履 滚滚洪流中&#xff0c;毁灭自己 ——24.2.18 数组常见算法 1.数组翻转 2.冒泡排序 3.二分查找 一、数组翻转 1.概述&#xff1a;数组对称索引位置上的元素互换 2.如何确定数组两端位置&#xff1f; int min…

vue2+高德地图web端开发(二)

前言&#xff1a; 高德地图输入提示与 POI 搜索相关文档&#xff1a;输入提示与 POI 搜索-服务插件和工具-进阶教程-地图 JS API 2.0 | 高德地图API (amap.com) 输入提示-输入提示-示例中心-JS API 2.0 示例 | 高德地图API (amap.com) 创建输入框&#xff1a; 引入Element组…

Selenium Grid分布式测试环境搭建

Selenium Grid简介 Selenium Grid实际上是基于Selenium RC的&#xff0c;而所谓的分布式结构就是由一个hub节点和若干个node代理节点组成。Hub用来管理各个代理节点的注册信息和状态信息&#xff0c;并且接受远程客户端代码的请求调用&#xff0c;然后把请求的命令转发给代理节…

猫头虎分享:一文带你搞懂什么是SaaS、PaaS、LaaS、CaaS、FaaS、MBaaS

博主猫头虎的技术世界 &#x1f31f; 欢迎来到猫头虎的博客 — 探索技术的无限可能&#xff01; 专栏链接&#xff1a; &#x1f517; 精选专栏&#xff1a; 《面试题大全》 — 面试准备的宝典&#xff01;《IDEA开发秘籍》 — 提升你的IDEA技能&#xff01;《100天精通鸿蒙》 …

拿捏c语言指针(中)

前言 书接上回 拿捏c语言指针&#xff08;上&#xff09; 此篇主要讲解的是指针与数组之间的爱恨情仇&#xff0c;跟着我的脚步一起来看看吧~ 创造不易&#xff0c;可以帮忙点点赞吗 如有差错&#xff0c;欢迎指出 理解数组名 数组名是首元素地址 例外 1.sizeof&#xff0…

《苍穹外卖》知识梳理P11-Apache POI导出报表

一.Apache POI 可以通过Apache POI处理excel文件&#xff0c;核心操作是读和写 应用场景 银行网银交易明细各种业务系统导出Excel报表批量导入业务数据 使用步骤 1.导入maven坐标 <dependency><groupId>org.apache.poi</groupId><artifactId>poi&…

萝卜大杂烩 | 把微信接入ChatGPT,变成聊天机器人竟然这么简单!(一起来尝试吧~)

本文来源公众号“萝卜大杂烩”&#xff0c;仅用于学术分享&#xff0c;侵权删&#xff0c;干货满满。 原文链接&#xff1a;把微信接入ChatGPT&#xff0c;变成聊天机器人竟然这么简单&#xff01; 最近的 ChatGPT 又再次火热起来了&#xff0c;各种周边工具也是层出不穷&…

力扣72. 编辑距离(动态规划)

Problem: 72. 编辑距离 文章目录 题目描述思路复杂度Code 题目描述 思路 由于易得将字符串word1向word2转换和word2向word1转换是等效的&#xff0c;则我们假定统一为word1向word2转换&#xff01;&#xff01;&#xff01; 1.确定状态&#xff1a;我们假设现在有下标i&#x…

SQL补充2:数据库的增操作

数据库的增操作 数据库的增操作主要涉及数据库的增加、数据表的增加、表记录增加以及表字段增加等&#xff1a; 数据库的增加非常简单&#xff0c;就是新创建一个数据库&#xff1b;表记录的增加指的就是新增表的数据行&#xff0c;可以是在已有表的基础上增加记录&#xff0…

菜刀HTTPTCP后门分析+防范

本文由掌控安全学院 - aj545302905 投稿 “菜刀”对于渗透测试者来说耳熟能详&#xff0c;但是大家用的菜刀真的安全吗&#xff1f;你能保证你所使用的工具不会被别人偷偷的塞入后门吗&#xff1f; 如果菜刀中被塞入后门 那我们岂不是成了别人的苦力。辛辛苦苦打下的shell就这…

简单介绍数据结构的基本概念

数据结构的基本概念 常用术语 数据 数据&#xff08;Data&#xff09;是客观事物的符号表示&#xff0c;是所有能输入到计算机中并被计算机程序处理的符号的总称。例如&#xff1a;整数、字符串、图形、图像、声音和动画等 数据元素 数据元素&#xff08;Data Element&…

基于SSM的宁夏旅游网站平台(有报告)。Javaee项目。ssm项目。

演示视频&#xff1a; 基于SSM的宁夏旅游网站平台&#xff08;有报告&#xff09;。Javaee项目。ssm项目。 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构&#xff0c;通过Spring …

解决updatexml和extractvalue查询显示不全

报错注入是一种常见的SQL 注入方式&#xff0c;通过注入代码&#xff0c;触发数据库的错误响应&#xff0c;并从错误信息中获取有用的信息。 updatexml和extractvalue updatexml和extractvalue 是常用的两个报错注入函数 http://localhost/sqli/Less-5/?id1%27and%20updat…

树莓派登录方式

目录 1.串口登录树莓派 1.1 USB-TTL连接树莓派串口 1.2 修改系统配置&#xff0c;启用串口登录树莓派 1.3 启动树莓派 2.网络方式登录树莓派 2.1 使树莓派接入网络 2.2 网络SSH 方式登录树莓派 2.2.1 打开ssh功能&#xff0c; 输入命令&#xff1a; 1.串口登录树莓派 1…