字符串函数-C语言

news2024/9/21 9:37:36

介绍

字符串函数,简单说,就是处理字符串的函数,头文件是string.h,以下是今天的讲解中会讲到的一系列字符串函数

  • 头文件:#include<string.h>
  • strlen:求字符串长度
  • strcpy:拷贝字符串
  • strcat:追加字符串
  • strcmp:比对字符串
  • strstr:一个字符串中找一个字符串
  • strncpy:拷贝字符串(拷贝指定字符串个数)
  • strncmp:比对字符串(比较指定个数字符个数)
  • strncat:追加字符串(追加指定个数个字符)
  • strtok:提取被分隔字符分隔字符串
  • strerror:可将参数部分错误码对应的错误信息返回

strlen

函数声明:size_t strlen(const char* str); 

功能:求字符串长度

使用条件:字符串以'\0'结尾

//函数自实现
#include <assert.h>
int my_strlen(const char* str)
{
	assert(str);
	int count = 0;
	while (*str != '\0') {
		count++;
		str++;
	}
	return count;
}

在库函数实现strlen时,是以'\0'为结尾判断标识的,如果在字符串中间混入了'\0',那么字符串长度的计算就会提前结束,计算的结果也会随之错误;反之,如果字符串结尾没有遇到'\0'那么在计时函数会一直顺着地址找下去,直到找到'\0'为止,这时候strlen返回的结果就是不可预测的了。

此处注明一下sizeof和strlen的区别,首先,sizeof是一个操作符,不是函数,其返回的值是数据类型所占内存大小,可处理的不只是字符串。而strlen恰相反,其只能用来处理字符串并计算其长度。

同时,也要注意到,strlen函数的返回值为size_t:无符号整型。

下面我举一个关于size_t易错的题

//问以下代码的输出结果
#include<stdio.h>
#include<string.h>
int main()
{
    if(strlen("abc") - strlen("abcd") > 0)
        printf("大于\n");
    else 
        printf("小于\n");
    return 0;
}

大家的第一反应是什么呢?我猜应该是输出小于。但仔细想一想,size_t作为无符号整型,相减会产生负数吗?不会的,所以最后的结果是打印大于。

//函数使用样例
#include<stdio.h>
#inlcude<string.h>
int main()
{
    char arr[] = "abcdef";
    int la = strlen(arr);
    printf("%s\n", la);//打印6
    return 0;
}

strcpy

函数声明:char* strcpy(char* destination, const char* source); 

功能:将参数source字符串拷贝至参数destination所指的地址

返回值:字符串destination起始地址

使用条件:源字符串必须以'\0',结尾

                目标空间必须大于等于拷贝过去字符所占的空间:包括'\0'

//自实现
#include<assert.h>
char* my_strcpy(char* str1, const char* str2)
{
	assert(str1 && str2);
	char* ret = str1;
	while (*str1++ = *str2++);
	*str1 = *str2;
	return ret;
}
//使用样例
#include<stdio.h>
#include<string.h>
int main() {
	char arr1[20] = "abcdef";
	char arr2[20] = { 0 };
	char* p1 = strcpy(arr2, arr1);
	printf("%s\n", arr2);//打印abcdef
	return 0;
}

strcat

函数声明:char* strcat(char* destination,const char* source);

功能:将source字符串追加到destination字符串末尾

返回值:destination字符串起始地址 

使用条件:两字符串均以'\0'结尾

                目标空间可被修改

                destination末尾'\0'会被覆盖

                目标空间足够大

注(追加位置):destination字符串'\0'及其后面

//自实现
char* my_strcat(char* dest, const char* src)
{
	char* ret = dest;
	while (*dest) dest++;
	while (*dest++ = *src++);
	return ret;
}
//样例使用
#inlude<stdio.h>
#inclue<string.h>
int main() {
	char arr1[20] = "abcdef";
    char arr2[] = "cd";
	printf("%s\n",strcat(arr1, arr2));//打印abcdefcd
	return 0;
}

 strcmp

函数声明:int strcmp(const char* str1,const char* str2);

功能:比较字符串

返回值:两字符串相同返回0

               前一个字符串在字典排序靠前返回负数

               后一个字符串在字典排序靠前返回正数

注:区分大小写比较,如果不希望区分可参考stricmp

//自实现
int my_strcmp(const char* s1, const char* s2)
{
	while (*s1 == *s2) {
		if (*s1 == '\0')return 0;
		s1++;
		s2++;
	}
	if (*s1 > *s2)return 1;
	else return -1;
}
//代码使用样例
#include<stdio.h>
#include<string.h>
int main() {
	char arr1[] = "abcdef";
	char arr2[] = "cd";
	printf("%d\n", strcmp(arr1, arr2));//打印一个负数
	return 0;
}

strstr

函数声明:char* strstr(const char* str1,const char* str2);

功能:检索str2在str1中第一次出现的位置

返回值: 返回子串str2在str1中第一次出现的位置,找不到返回空指针

//自实现
char* my_strstr(const char* str1,const char* str2)
{
	const char* cur = str1;
	const char* s1 = NULL;
	const char* s2 = NULL;
	if (!*str2)return (char*)str1;
	while (*cur) {
		s1 = cur;
		s2 = str2;
		while (*s1 == *s2) {
			s1++;
			s2++;
		}
		if (*s2 == '\0')return (char*)cur;
		cur++;
	}
	return NULL;
}

以上自实现代码是一份暴力查找版,算法和时间复杂度并不像库函数中那样是最优的,如果想了解真正的内部实现可以去了解KMP算法,此代码只是想让读者更容易了解此函数的功能。

//使用样例
#include<stdio.h>
#include<string.h>
int main() {
	char arr1[20] = "abcdef";
	char arr2[] = "cd";
	printf("%s\n",strstr(arr1, arr2));//打印cdef
	return 0;
}

strncpy 

 函数声明:char* strncpy(char* destination,const char* source,size_t mun);

功能:与strcpy很相似,只是多累个参数,拷贝指定个数的字符串

返回值:字符串destination起始地址

使用条件:源字符串必须以'\0',结尾

                目标空间必须大于等于拷贝过去字符所占的空间:包括'\0'

注:在拷贝完指定个数字符后,不会在串的末尾再拷贝'\0'

//自实现
#include<assert.h>
char* my_strcpy(char* str1, const char* str2,size_t num)
{
	assert(str1 && str2);
	char* ret = str1;
	while ((*str1++ = *str2++)&&num!=0) num--;
	*str1 = *str2;
	return ret;
}

strncmp

函数声明:int strcmp(const char* str1,const char* str2,size_t num);

功能:和strcmp很相似,但是多了个参数,比较的是指定个数的字符

返回值:两字符串相同返回0

               前一个字符串在字典排序靠前返回负数

               后一个字符串在字典排序靠前返回正数

注:返回值描述的字符串比较的比较是用num个数截掉后的字符串,具体可参考自实现

//自实现
#include<assert.h>
int my_strncmp(const char* arr1, const char* arr2, size_t num)
{
	assert(arr1 && arr2);
	for (int i = 0; i < num; i++) {
		if (*(arr1 + i) == *(arr2 + i));
		else if (*(arr1 + i) > *(arr2 + i)) return 1;
		else return -1;
	}
	return 0;
}
//使用样例
#include<stdio.h>
#include<string.h>
int main()
{
	char arr1[] = "abcdefg";
	char arr2[] = "abcdfea";
	int ret =strncmp(arr1, arr2, 4);
	printf("%d\n", ret);//打印0
	ret = strncmp(arr1, arr2, 5);
	printf("%d\n", ret);//打印一个负数
	return 0;
}

strncat

函数声明: char* strcat(char* destination,const char* source,size_t num);

功能:与strcat功能相似,将source指定个数的字符串追加到destination字符串末尾

返回值:destination字符串起始地址 

使用条件:两字符串均以'\0'结尾

                目标空间可被修改

                destination末尾'\0'会被覆盖

                目标空间足够大

注:destination字符串'\0'及其后面

        追加num个字符后会自动添加'\0'        

//自实现
#include<assert.h>
char* my_strncat(char* arr1, const char* arr2, size_t num)
{
	assert(arr1 && arr2);
	char* ret = arr1;
	while (*arr1)arr1++;
	for (int i = 0; i < num; i++) {
		*(arr1 + i) = *(arr2 + i);
	}
	*(arr1 + num) = '\0';
	return ret;
}
//使用样例
#include<stdio.h>
#include<string.h>
#include<assert.h>
int main()
{
	char arr1[20] = "abcdefg";
	char arr2[20] = "rhmotyi";
	char* pc = strncat(arr1, arr2, strlen(arr2));
	printf("%s\n", pc);    //打印abcdefgrhmotyi
	pc = strncat(arr1, arr2, strlen(arr2) - 2);
	printf("%s\n", pc);    //打印abcdefgrhmotyirhmot
	return 0;
}

strtok

函数声明:char* strtok(char* str, const char* sep);

功能:提取被分隔字符分隔字符串

           其中:sep里面放的是分隔符;str里面放的是带分隔符的字符串

           调用一次,将最首先找到的分隔符变成'\0',并标记下一个位置

返回值:上一次标记位置(如第一次调用则返回str首元素地址)

               如果字符串中不存在更多标记,则返回NULL

注:strtok第一个参数不为NULL时,函数找到str中第一个标记,保存在其字符串中的位置

       strtok第一个参数为NULL时,函数将从被保存的位置开始,寻找下一个标记

       strtok内部其实存在静态变量,运行结束后上一次运行数据会有存留(也就是标记地址)

//使用样例
#include <stdio.h>
#include <string.h>
int main()
{
	const char* p = "@.";
	char arr[] = "zhangsan@1623.com";
	char* s = strtok(arr, p);
	printf("%s\n", s);//zhangsan
	s = strtok(NULL, p);
	printf("%s\n", s);//1623
	s = strtok(NULL, p);
	printf("%s\n", s);//com
	return 0;
}

但是上面的代码似乎有点冗杂,你完全可以写在循环中,见代码

//使用样例(循环版)
#include <stdio.h>
#include <string.h>
int main()
{
	const char* p = "@.";
	char arr[] = "zhangsan@1623.com";
	//char* s = strtok(arr, p);
	//printf("%s\n", s);//zhangsan
	//s = strtok(NULL, p);
	//printf("%s\n", s);//1623
	//s = strtok(NULL, p);
	//printf("%s\n", s);//com
	for (char* s = strtok(arr, p); s != NULL; s = strtok(NULL, p))
		printf("%s\n", s);
	return 0;
}

这两份代码打印出来的结果是相同的

strerror

函数声明:char* strerror(int errnum);

功能:可以将参数部分错误码对应的错误信息字符串返回 

 C语言规定了一些错误码,放在errno.h这个头文件中说明

//样例使用
#include <stdio.h>
#include <string.h>
int main()
{
	for (int i = 0; i < 10; i++)
		printf("%d:%s\n",i, strerror(i));
	return 0;
}

下面是输出结果

 结语

那么本篇博客到这里就要结束了,我们介绍了大部分在平时coding中可能运用到的字符串函数,以及一些函数的自实现,大家可以看看,更利于对函数的了解。如果觉得这篇博客对你有帮助的话,还请点个小赞收藏一下再走啊,如果本篇博客有任何错误也欢迎在评论区讨论或者私我哦--比心♥

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

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

相关文章

物联网实训室虚拟仿真软件建设方案

一、概述 物联网实训室虚拟仿真软件旨在紧密围绕立德树人的根本任务&#xff0c;充分依托先进的数字技术&#xff0c;并对接物联网行业的发展趋势和人才需求。通过对比真实企业工作环境&#xff0c;融合创新创业教育基因&#xff0c;秉承虚拟仿真技术与教育教学深度融合的理念&…

艺海泛舟——尹星从艺六十年作品展暨学术交流首展作品(二)

简历&#xff1a; 1944年1月25日出生于山西省阳高县&#xff0c;内蒙古师范学院艺术系美术专业&#xff0c;师从水彩之父李剑晨&#xff0c;北京京华美术学院创立者邱石冥&#xff0c;徐坚。与吴冠中&#xff0c;朱德群&#xff0c;赵无极&#xff0c;杨飞云是同门。擅长&…

解决CentOS下PHP system命令unoconv转PDF提示“Unable to connect or start own listener“

centos系统下&#xff0c;用php的system命令unoconv把word转pdf时提示Unable to connect or start own listene的解决办法 unoconv -o /foo/bar/public_html/upload/ -f pdf /foo/bar/public_html/upload/test.docx 2>&1 上面这个命令在shell 终端能执行成功&#xff0c…

优测云服务平台- 探索性测试策略分享

版权声明&#xff1a;本文作者 优测团队测试专家 郑凯泽 南明玮。 探索性测试是对常规的系统测试、新需求测试及专项测试的重要补充&#xff0c;往往能在短时间内发现更多的问题&#xff0c;一起来看看优测测试专家的分享吧~ 一、背景 优测团队长期承接腾讯社交产品、办公产…

​在做接口测试的时候,如果接口还没有开发好,你这边应该怎么去介入测试?

&#x1f4e2;专注于分享软件测试干货内容&#xff0c;欢迎点赞 &#x1f44d; 收藏 ⭐留言 &#x1f4dd; 如有错误敬请指正&#xff01;&#x1f4e2;交流讨论&#xff1a;欢迎加入我们一起学习&#xff01;&#x1f4e2;资源分享&#xff1a;耗时200小时精选的「软件测试」资…

机械臂仿真之vrep如添加视觉传感器

基于视觉的机械臂作业任务&#xff0c;如何在vrep中加入视觉传感器&#xff0c;并获取画面&#xff1f;

【elementUI】el-tab相关问题

Tabs 标签页 分隔内容上有关联但属于不同类别的数据集合。 <template><el-tabs v-model"activeName" tab-click"handleClick"><el-tab-pane label"用户管理" name"first">用户管理</el-tab-pane><el-tab-p…

什么是云安全?如何制定云安全战略

云计算允许组织通过互联网按需向其客户、合作伙伴或员工提供关键业务应用程序、服务和资源&#xff0c;换句话说&#xff0c;不再需要物理维护资源&#xff0c;每当通过 Internet 从计算机访问文件或服务时&#xff0c;都是在访问云。 迁移到云可以帮助企业增强安全性、简化运…

zemax之初级像差理论与像差校正——球差

一、初级像差理论 1.1厚透镜初级像差 由于结构和机械强度的需要&#xff0c;任何光学透镜都具有一定厚度。对于正透镜其边缘厚度一般不应小于3mm&#xff1b;对于负透镜&#xff0c;中心厚度不应小于透镜孔径1/10~1/15&#xff0c;以防止安装和固定变形。 除此之外透镜的厚度…

为什么要在项目中使用TypeScript?

随着越来越多的开发人员采用TypeScript&#xff0c;人们需要了解在下一个项目中应该使用TypeScript的原因。尽管它在早期应用中遇到了一些阻力&#xff0c;但在过去十年&#xff0c;它迅速成为一种广泛使用的编程语言。 以下介绍如何使用TypeScript以及它给开发人员带来的一些好…

每日一练 | 华为认证真题练习Day140

1、如图所示&#xff0c;网络管理员希望将SWA与SWB之间的两条物理链路手工聚合成一条Eth-trunk链路&#xff1b;下列描述正确的是&#xff08;&#xff09;。 A. 不能被聚合 B. 聚合后可以正常工作 C. 可以聚合&#xff0c;聚合后只有GE端口能收发数据 D. 可以聚合&#xff…

选择合适的企业邮箱供应商的策略与技巧

如果您的公司是一个业务遍布全球的组织&#xff0c;选择合适的企业邮箱供应商将是一个重要的决策。 以下是在选择全球适用的企业邮箱供应商时应考虑的关键因素&#xff1a; 全球覆盖和数据中心位置&#xff1a;确保企业邮箱供应商在您业务遍及的国家和地区均具备良好的服务覆盖…

功能全面又强大的同步备份软件,你找到了吗?

随着企业规模的不断扩大&#xff0c;许多企业都会拥有自己的数据中心。因此每日员工都需要在服务器与服务中心之间调取文件&#xff0c;同时还需要对每日新增的业务数据进行实时同步。如果量比较小&#xff0c;一般问题不大&#xff1b;一旦数据比较大&#xff0c;量也比较大&a…

(Ant X6)子组件里的流程图画布无法显示

(Ant X6)子组件里的流程图画布无法显示 问题背景&#xff1a;侧导航页面都是子组件,建模页面的画布无法显示 解决前&#xff1a; 解决后&#xff1a; 解决思路&#xff1a;点击建模菜单时再次加载对应组件 在 Vue 中&#xff0c;每个组件都有一个唯一的 key 属性。当组件的 ke…

强化学习-DQN

网上看来很多&#xff0c;但是还是觉得这篇文章将得最好&#xff1a; 可视化强化学习解释 - Deep Q Networks&#xff0c;循序渐进 |Ketan Doshi 博客 (ketanhdoshi.github.io)

VirtualBox 7.0.8(虚拟机软件)

VirtualBox是一款开源的虚拟机软件&#xff0c;它是使用Qt编写&#xff0c;在Sun被Oracle收购后正式更名成Oracle VM VirtualBox。它可以在VirtualBox上安装并且执行Solaris、Windows、DOS、Linux、OS/2 Warp、BSD等系统作为客户端操作系统。使用者可以在VirtualBox上安装并且运…

Maven下载与安装教程

一、下载 Maven 进入 Maven 官网&#xff1a;maven.apache.org/download.cgi 选择 .zip 文件下载&#xff0c;最新版本是 3.9.5 二、安装 Maven 将 .zip 文件解压到没有中文没有空格的路径下。例如下图&#xff0c;在创建一个repository的空文件夹在他的下面&#xff0c;用于…

增强静态数据的安全性

静态数据是数字数据的三种状态之一&#xff0c;它是指任何静止并包含在永久存储设备&#xff08;如硬盘驱动器和磁带&#xff09;或信息库&#xff08;如异地备份、数据库、档案等&#xff09;中的数字信息。 静态数据是指被动存储在数据库、文件服务器、端点、可移动存储设备…

分享86个简历竞聘PPT,总有一款适合您

分享86个简历竞聘PPT&#xff0c;总有一款适合您 86个简历竞聘PPT下载链接&#xff1a;https://pan.baidu.com/s/130iX0EIH6J-PFzb6HcntcQ?pwd8888 提取码&#xff1a;8888 Python采集代码下载链接&#xff1a;采集代码.zip - 蓝奏云 学习知识费力气&#xff0c;收集整…

uniapp 连接斑马PDA调试

1、先把PDA设置成开发者模式 打开设置--》关于手机 --》单击版本号&#xff0c;5次以上 连线单击5次以上 2、后退--》找到系统 --》高级 3、打开 --》开发都模式 4、找到调试 --》 打开USB调试 5、设置USB偏好设置&#xff0c;插入电脑连接PDA就会在通知栏上显示&#xff0c;默…