字符函数和字符串函数(C语言进阶)

2023-09-21 22:04:18

前言
C语言中对字符和字符串的处理很是频繁,但是C语言本身是没有字符串类型的,字符串通常放在常量字符串中或者字符数组中。
字符串常量适用于那些对它不做修改的字符串函数

一.求字符串长度

1.strlen

size_t strlen ( const char * str );

在这里插入图片描述

代码演示:
#include <string.h>

int main()
{
	const char* str = "abcdef";
    //"abcdef"==str
	size_t len1 = strlen("abcdef");
	size_t len2 = strlen(str);

	printf("%d\n", len1);
	printf("%d\n", len2);

	return 0;
}

运行结果:
在这里插入图片描述

1.字符串已经 ‘\0’ 作为结束标志,strlen函数返回的是在字符串中 ‘\0’ 前面出现的字符个数(不包含 ‘\0’ )。
在这里插入图片描述

2.参数指向的字符串必须要以 ‘\0’ 结束。
数组中不含‘\0’,所以计算strlen为随机值
在这里插入图片描述

3.注意函数的返回值为size_t,是无符号的( 易错 )
让我们来证明下strlen的类型为size_t(无符号整型),看以下代码:

int main()
{
	//两个无符号数相减得到的还是无符号数
	// 3 - 6 = -3
	// -3
	//10000000000000000000000000000011 原码
	//11111111111111111111111111111100 反码
	//11111111111111111111111111111101 补码
	
	if (strlen("abc") - strlen("abcdef") > 0)
		printf(">=\n");
	else
		printf("<\n");
	return 0;
}

让我们看看运行结果:
在这里插入图片描述

对运行结果进行分析
在这里插入图片描述

若想比较两个strlen算出的实际大小,将strlen强转成int类型如下

代码演示:
#include<stdio.h>
#include <string.h>
int main()
{
	if ((int)strlen("abc") - (int)strlen("abcdef") > 0)
		printf(">=\n");
	else
		printf("<\n");
	return 0;
}

运行结果:
在这里插入图片描述

模拟实现strlen

1. 计数器
#include<stdio.h>
int my_strlen(const char* str)
{
	int count = 0;
	while (*str != '\0')
	{
		str++;
		count++;
	}
	return count;
}
int main()
{
	int ret = 0;
	char arr[] = "abcdef";
	ret = my_strlen(arr);
	printf("%d\n", ret);
}

运行结果:
在这里插入图片描述

2. 指针-指针
#include<stdio.h>
int my_strlen(const char* str)
{
	const char* p = str;
	while (*str != '\0')
	{
		str++;
	}
	return str - p;
}
int main()
{
	int ret = 0;
	char arr[] = "abcdef";
	ret = my_strlen(arr);
	printf("%d\n", ret);
}

运行结果:
在这里插入图片描述

3. 递归
#include<stdio.h>
int my_strlen(const char* str)
{
	if (*str == '\0')
		return 0;
	else
		return 1 + my_strlen(str + 1);
}
int main()
{
	int ret = 0;
	char arr[] = "abcdef";
	ret = my_strlen(arr);
	printf("%d\n", ret);
}

运行结果:
在这里插入图片描述

二.长度不受限制的字符串函数介绍

1.strcpy

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

在这里插入图片描述

代码案例:
#include<stdio.h>
#include<string.h>
int main()
{
	char arr1[20] = { 0 };
	char arr2[] = "Hello";
	strcpy(arr1, arr2);
	printf("%s\n", arr1);
	return 0;
}

运行结果:
在这里插入图片描述

1.源字符串必须以 ‘\0’ 结束。
对比以下两张图:
没有‘\0’:

有‘\0’:
在这里插入图片描述

2.会将源字符串中的 ‘\0’ 拷贝到目标空间。
在这里插入图片描述

3.目标空间必须足够大,以确保能存放源字符串。
在这里插入图片描述

4.目标空间必须可变。
在这里插入图片描述

模拟实现strcpy

#include<stdio.h>
#include<assert.h>
char* my_strcpy(char* dest, const char* src)
{
	
	char* ret = dest;
	assert(dest && src);
	while (*dest++ = *src++)
	{
		;
	}
	return ret;
}
int main()
{
	char arr1[20] = { 0 };
	char arr2[] = "abcd";
	printf("%s\n", my_strcpy(arr1,arr2));
	return 0;
}

运行结果:
在这里插入图片描述

2.strcat

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

在这里插入图片描述

代码演示:
int main()
{
	char arr1[20] = "abc";
	char arr2[] = "def";
	strcat(arr1, arr2);
	printf("%s\n", arr1);
	return 0;
}

运行结果:
在这里插入图片描述
1.源字符串必须以 ‘\0’ 结束。(保证能找到目标空间末尾)
在这里插入图片描述
在这里插入图片描述
2.原字符串也必须有‘\0’,再拷贝时将原字符串‘\0’拷贝过去
在这里插入图片描述

3.目标空间必须有足够的大,能容纳下源字符串的内容,目标空间必须可修改。
在这里插入图片描述

4.字符串自己给自己追加,如何?

代码结果:
int main()
{
	char arr1[20] = "abc";
	strcat(arr1, arr1);
	printf("%s\n", arr1);
	return 0;
}

运行结果:
在这里插入图片描述

模拟实现strcpy

代码案例:
#include<stdio.h>
#include<string.h>
#include<assert.h>
char* my_strcat(char* dest, const char* src)
{
	char* ret = dest;
	assert(dest && src);
	//1. 找到目标空间的末尾
	while (*dest != '\0')
	{
		dest++;
	}
	//2. 数据追加
	while (*dest++ = *src++)
	{
		;
	}
	return ret;
}
int main()
{
	char arr1[20] = "abc";
	char arr2[] = "def";
	my_strcat(arr1, arr2);
	printf("%s\n", arr1);
	return 0;
}

运行结果:
在这里插入图片描述
注:这种模拟实现不可以完成自身追加
在这里插入图片描述
为什么不能实现自身追加呢?
在这里插入图片描述

3.strcmp

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

在这里插入图片描述

标准规定:

第一个字符串大于第二个字符串,则返回大于0的数字
第一个字符串等于第二个字符串,则返回0
第一个字符串小于第二个字符串,则返回小于0的数字

模拟实现:

//代码演示:
#include<stdio.h>
#include<string.h>
#include<assert.h>
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;
}

int main()
{
	char arr1[] = "abq";
	char arr2[] = "abc";
	if (my_strcmp(arr1, arr2) > 0)
	{
		printf(">\n");
	}
	else
	{
		printf("<=\n");
	}

	return 0;
}

运行结果:
在这里插入图片描述
💘不知不觉,字符函数和字符串函数1(C语言进阶)以告一段落。通读全文的你肯定收获满满,不久的将来会继续更新字符函数和字符串函数,让我们继续为C语言学习共同奋进!!!

更多推荐

python/C++二分查找库函数(lower_bound() 、upper_bound,bisect_left,bisect_right)

二分查找是一种经典的搜索算法,广泛应用于有序数据集中。它允许在大型数据集中高效地查找目标元素,减少了搜索的时间复杂度。本文将介绍在C++和Python中内置的二分查找函数,让二分查找变得更加容易。c++lower_bound()、upper_bound定义在<algorithm>头文件中,lower_bound和upp

大数据分布式处理框架Hadoop

大数据是什么大数据容量常以TB、PB、甚至EB为单位,远超传统数据库的承载能力,无论入库还是查询都出现性能瓶颈。Hadoop是什么Hadoop是开源的分布式计算技术框架,用于处理大规模数据和实现分布式存储。Hadoop核心组件HDFS(HadoopDistributedFileSystem分布式文件系统):是Hadoo

学习Nano编辑器:入门指南、安装步骤、基本操作和高级功能

文章目录使用Nano编辑器入门指南引言1.1关于Nano编辑器1.2Nano的起源和特点安装Nano2.1在Debian/Ubuntu系统上安装Nano2.2在CentOS/RHEL系统上安装Nano2.3在其他Linux发行版上安装Nano启动Nano3.1命令行启动Nano3.2打开文件Nano的基本操作4.1光标

如何批量为文件夹命名

如果你想要命名一些这样名字具有规律性的文件夹,当文件的数量增多,一个一个命名是非常耗费时间的。很容易想到,如果使用EXCEL,只需往下拉,就能很轻松的拉出1到5。那么,我们如何利用EXCEL来对文件夹进行快速的批量命名呢?以上图为例子,人名可能是我们已知的,可以从表格直接复制过来。而客户名1到客户名5,我们可以直接使用

大数据Flink(六十四):Flink运行时架构介绍

文章目录Flink运行时架构介绍一、系统架构二、​​​​​​​​​​​​​​整体构成三、作业管理器(JobManager)四、任务管理器(TaskManager)Flink运行时架构介绍我们已经对Flink的主要特性和部署提交有了基本的了解,那它的内部又是怎样工作的,集群配置设置的一些参数又到底有什么含义呢?接下来我们

基于vue的黑马前端项目小兔鲜

目录项目学习初始化项目建立项目引入elementpluselementPlus主题设置配置axios路由引入静态资源自动导入scss变量Layout页组件结构快速搭建字体图标渲染一级导航渲染吸顶导航交互实现Pinia优化重复请求Home页分类实现banner轮播图新鲜好物实现人气推荐实现懒加载指令实现产品列表实现Goo

正确设置PyTorch训练时使用的GPU资源

背景:最近在使用HuggingFace的transformersapi来进行预训练大模型的微调,机器是8卡的GPU,当我调用trainer.train()之后,发现8个GPU都被使用了,因为这个机器上面还有其他人跑的模型,当我进行训练的时候,GPU的使用率就接近了100%,这样导致其他人的模型响应速度特别的慢,因为他们

微信小程序如何刷新当前页面

微信小程序是一种快速发展的移动应用程序开发平台,它提供了许多功能和特性,使开发者能够轻松创建功能丰富的小程序。在开发小程序时,有时我们需要刷新当前页面来更新数据或重新加载页面内容。本文将解释如何在微信小程序中刷新当前页面的代码。引言微信小程序的流行使得越来越多的开发者将其作为构建移动应用的首选平台。然而,与传统的网页开

快速了解Apipost

随着数字化转型的加速,API(应用程序接口)已经成为企业间沟通和数据交换的关键。而在API开发和管理过程中,API文档、调试、Mock和测试的协作显得尤为重要。Apipost正是这样一款一体化协作平台,旨在解决这些问题,提高API开发效率和质量。Apipost提供API文档管理功能,让后端开发人员可以在开发完接口调试的

cookie信息无法获取问题研究

背景在oneapi这个前后端都有的开源项目中,我想接入chatnextweb到oneapi的后端。由于需要二开chatnextweb,添加登录注册功能,考虑到java后端的性能问题和内存占用,决定不重启写个服务,而是将登录注册接入到oneapi的登录注册api,没错,和oneapi的前端公用一套登录注册熬了个大夜后,终

GLTF编辑器:在线模型材质编辑工具

GLTF编辑器是一个功能强大、易于使用的在线3D模型编辑和查看工具,它支持多种格式的3D模型导入并将模型导出为GLB格式,除了可以对3D模型进行基本属性的修改之外,还支持对模型原点重置以及模型材质纹理修改。对于3D开发者和设计师来说,GLTF编辑器是一个非常有用的工具,可以帮助他们更方便地处理3D模型数据。1、首先介绍

热文推荐