【C语言基础】:字符函数和字符串函数

news2024/11/18 21:27:59

文章目录

    • 一、字符函数
      • 1. 字符分类函数
      • 2. 字符转化函数
    • 二、字符串函数
      • 1. strlen函数的使用和模拟实现
        • strlen函数的使用
        • strlen函数的模拟实现
      • 2. strcpy函数的使用和模拟实现
        • strcpy函数的使用
        • strcpy函数的模拟实现
      • 3. strcat函数的使用和模拟实现
        • strcat函数的使用
        • strcat函数的模拟实现
      • 4. strcmp函数的使用和模拟实现
        • strcmp函数的使用
        • strcmp函数的模拟实现

一、字符函数

1. 字符分类函数

C语言中有一系列的函数是对字符进行分类的,就是对判断一个字符属于什么类型的字符,这类字符函数的使用都要包含一个头文件ctype.h

在这里插入图片描述
这些函数的使用方法非常类似,这里我们就只举一个例子。

 int islower ( int c );

在C语言中,islower是一个用于判断字符是否为小写字母的函数。该函数接受一个整数参数c,它应该是无符号字符或EOF(常量)。如果参数c是小写字母,则返回非零值(真);否则返回0(假)。

【示例】将字符串中的小写字母转大写,其他字符不变。

#include<stdio.h>
#include<ctype.h>
int main()
{
	char str[] = "Test String.\n";
	int i = 0;
	char c;
	while (str[i])
	{
		c = str[i];
		if (islower(c))
			c -= 32;  // 小写字母的ASCII码值减去32就等于它对应的大写字母的ASCII码
		putchar(c);
		i++;
	}
	return 0;
}

在这里插入图片描述

2. 字符转化函数

C语言提供了2个字符转换函数:

int tolower ( int c ); //将参数传进去的⼤写字⺟转⼩写 
int toupper ( int c ); //将参数传进去的⼩写字⺟转⼤写

在上面的示例中,我们将小写转大写,是-32完成的效果,有了转换函数,就可以直接使用 tolower 函
数。

#include<stdio.h>
#include<ctype.h>
int main()
{
	char str[] = "Test String.\n";
	int i = 0;
	char c;
	while (str[i])
	{
		c = str[i];
		if (islower(c))
			c = toupper(c);
		putchar(c);
		i++;
	}
	return 0;
}

在这里插入图片描述

二、字符串函数

1. strlen函数的使用和模拟实现

函数原型:

size_t strlen ( const char * str );
  • 字符串以 ‘\0’ 作为结束标志,strlen函数返回的是在字符串中 ‘\0’ 前⾯出现的字符个数(不包含 ‘\0’ )。
  • 参数指向的字符串必须要以 ‘\0’ 结束。
  • 注意函数的返回值为 size_t,是无符号的( 易错 )
  • strlen的使用需要包含头文件
  • 学会strlen函数的模拟实现
strlen函数的使用

计算字符串的长度

#include<stdio.h>
#include<string.h>
int main()
{
	char str1[] = "abcdef";
	size_t len = strlen(str1);
	printf("%zu\n", len);
	return 0;
}

在这里插入图片描述

strlen函数的模拟实现

方法一:计数器的方式实现

#include<stdio.h>
#include<assert.h>
int my_strlen(const char* str)  // const修饰,使其不能被修改
{
	int count = 0;
	assert(str);  // 断言,避免传入空指针
	while (*str)
	{
		count++;  // 进入循环,说明不为空
		str++;
	}
	return count;
}
int main()
{
	char str1[] = "abcdef";
	int len = my_strlen(str1);
	printf("%d\n", len);
	return 0;
}

在这里插入图片描述
方法二:递归实现

#include<stdio.h>
#include<assert.h>
int my_strlen(const char* str1)
{
	assert(str1);   // 断言,避免传入空指针
	if (*str1 == '\0')
		return 0;  // 等于'\0',说明字符串结束,直接返回
	else
		return 1 + my_strlen(str1 + 1);
}
int main()
{
	char str1[] = "abcdef";
	int len = my_strlen(str1);
	printf("%d\n", len);
	return 0;
}

在这里插入图片描述
方法三:指针 - 指针的方式实现

#include<stdio.h>
int my_strlen(char* s)
{
	char* p = s;
	while (*p != '\0')
		p++;
	return p - s;
}
int main()
{
	int len = my_strlen("abcdef");
	printf("%d\n", len);
	return 0;
}

在这里插入图片描述

2. strcpy函数的使用和模拟实现

函数原型:

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

将source指向的C字符串复制到destination指向的数组中,包括结束的null字符(并在该点停止)。

strcpy函数的使用

将str1中的内容拷贝到str2中去

#include<stdio.h>
#include<string.h>
int main()
{
	char str1[] = "hello world";
	char str2[20] = { 0 };
	strcpy(str2, str1);  // 将str1中的内容拷贝到str2中去
	printf("%s\n", str2);
	return 0;
}

在这里插入图片描述
注意:这里str2的空间必须得能够装下str1中的内容,也就是str2的空间要足够大

strcpy函数的模拟实现

在模拟实现之前,首先我们要了解strcpy的实现原理,只有这样才会更加方便我们去模拟实现。那么,strcpy的实现原理是怎样的呢?
这里我们根据上面的代码进行修改再调试一下:

#include<stdio.h>
#include<string.h>
int main()
{
	char str1[] = "hello world";
	char str2[20] = "xxxxxxxxxxxxxxxxxx";
	strcpy(str2, str1);
	printf("%s\n", str2);
	return 0;
}

str1加上空格和字符串的结束标志 ‘\0’ 一共有12个字符,如果在str2中第12(下标为11)个字符也变成了 ‘\0’,那就说明是把str1中的 ‘\0’ 搬到了str2中。
在这里插入图片描述
事实证明strcpy的确是这样实现字符串拷贝的,所以我们在模拟实现时要一直将字符串拿到 ‘\0’ 才能结束。
注意

  • 源字符串必须以 ‘\0’ 结束。
  • 会将源字符串中的 ‘\0’ 拷贝到目标空间。
  • 目标空间必须足够大,以确保能存放源字符串。
  • 目标空间必须可修改。

接下来我们来一步一步模拟strcpy函数

void my_strcpy(char* dest, char* src)
{
	while (*src != '\0')  // 拷贝'\0'之前的内容
	{
		*dest = *src;
		dest++;
		src++;
	}
	*dest = *src;  // 拷贝'\0'
}

这样其实也能模拟实现,但这样的代码明显还有很多缺陷,还有很多优化的空间。

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

这是最终优化后的代码,大家看到这里可能会有点懵,别急,我们慢慢来:

  1. 首先为了确保安全,我们得判断传入进来的dest和src是否为空指针,所以这里要断言一下,一旦他们为空指针就直接报错。
  2. 其次,为了确保src不被修改,可以用const进行修饰。在while循环*dest++ = *src++作为判断条件,这里因为++的优先级高于 * ,但他是后置++,先使用再自增,所以也就是先解引用再++。
  3. 第一次判断会将src中的第一个字符给dest,因为while循环里的是空语句,执行了也不会有任何效果,然后就这样一直判断,一直将src中的字符给到dest,知道将’\0’给到dest,这时whlie循环括号里的表达式为0,跳出循环,但src中包括’\0’全部给到了dest。
  4. 因为原strcpy函数的返回值是char * 类型的,我们这里为了模拟也改成char * 类型的,原本我们是要返回dest的,但由于在while循环里面dest已经后置++不在指向首地址,所以在这之前我们先用一个指针ret来存放dest的首地址,最后我们直接返回ret就行了。
#include<assert.h>
#include<stdio.h>
#include<ctype.h>
char* my_strcpy(char* dest, const char* src)
{
	assert(dest != NULL);
	assert(src != NULL);
	char* ret = dest;
	while (*dest++ = *src++)
		;
	return ret;
}
int main()
{
	char str1[] = "hello world";
	char str2[20] = "xxxxxxxxxxxxxxxxxx";
	my_strcpy(str2, str1);
	printf("%s\n", str2);
	char * ret = my_strcpy(str2, str1);
	printf("%s\n", ret);
	return 0;
}

在这里插入图片描述

3. strcat函数的使用和模拟实现

函数原型:

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

连接字符串
将原字符串的副本追加到目标字符串。destination中的结束null字符被source的第一个字符覆盖,并且在destination中由两者串联形成的新字符串的末尾包含一个空字符。

  • 源字符串必须以 ‘\0’ 结束。
  • ⽬标字符串中也得有 \0 ,否则没办法知道追加从哪⾥开始。
  • ⽬标空间必须有⾜够的⼤,能容纳下源字符串的内容。
  • ⽬标空间必须可修改。
  • 字符串⾃⼰给⾃⼰追加,如何?
strcat函数的使用

将arr2中的字符串追加到arr2后面去。

#include<stdio.h>
#include<string.h>
int main()
{
	// 注意数组空间大小,要足以容纳追加后的字符数量
	char arr1[20] = "hello ";  
	char arr2[] = "world";
	strcat(arr1, arr2);
	printf("%s\n", arr1);
	return 0;
}

在这里插入图片描述
注意:这里arr1的空间必须得能够装下arr2追加到arr1后中的内容,也就是arr1的空间要足够大。

strcat函数的模拟实现

和之前一样,在模拟实现之前我们先要了解strcat函数的实现原理。这里我们同样调试一下看看。
在这里插入图片描述
调试之后可以看到,追加的过程是从arr1中的 \0开始追加,但是arr2中的 \0并没有拷贝到arr1中。
根据strcat函数的实现原理我们来进行模拟实现:

#include<stdio.h>
#include<assert.h>
char* my_strcat(char* dest, const char* src)
{
	assert(dest && src);  // 断言判断传入的是否为空指针
	char* ret = dest;  // 记录dest的起始地址
	// 找到目标空间的\0
	while (*dest != '\0')
		dest++;
	// 拷贝追加
	while (*dest++ = *src++)
		;
	return ret;
}
int main()
{
	// 注意数组空间大小,要足以容纳追加后的字符数量
	char arr1[20] = "hello ";  
	char arr2[] = "world";
	// my_strcat(arr1, arr2);
	char* ret = my_strcat(arr1, arr2);
	printf("%s\n", arr1);
	printf("%s\n", ret);
	return 0;
}

在这里插入图片描述
和上面一样,调试之后我们知道被追加的字符串从 \0开始,但追加的字符串并不会将 \0给追加进来。

  1. 首先我们肯定要判断传入进来的指针是否为空指针,这里要断言一下,而且为了防止要追加的字符串被修改,我们要将它用const修饰一下。
  2. 因为被追加的字符串是从 \0开始追加的,所以我们在追加之前要将指针指向字符串结尾,这里用while循环判断,只要不是\0我们就加一,直到指向字符串结尾。
  3. 这里就和strcpy的模拟一样了,也是在while循环的条件那里判断*dest++ = *src++,只要 *src没到\0,那么这个表达式的结果就不会为0,循环就还会继续,知道\0为止。
  4. strcat函数原型的返回值的char * 类型的,这里我们也返回char * 类型的,但要注意这里返回的是被追加字符串的起始地址,由于我们在实现追加的过程中将指针往后进行了偏移不在指向起始地址,所以我们要创建一个指针变量来记录被追加字符串的起始地址,最后返回这个创建的指针变量就行了。

4. strcmp函数的使用和模拟实现

函数原型:

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

比较两个字符串
比较C字符串str1和C字符串str2。这个函数执行字符的二进制比较。
这个函数开始比较每个字符串的第一个字符。如果它们彼此相等,则继续执行以下对,直到字符不同或达到终止空字符为止。

  • 标准规定:
    ◦ 第⼀个字符串大于第⼆个字符串,则返回大于0的数字
    ◦ 第⼀个字符串等于第⼆个字符串,则返回0
    ◦ 第⼀个字符串小于第⼆个字符串,则返回小于0的数字
    ◦ 那么如何判断两个字符串? 比较两个字符串中对应位置上字符ASCII码值的大小。
strcmp函数的使用

比较三个字符串

#include<stdio.h>
#include<string.h>
int main()
{
	char arr1[] = "abcdef";
	char arr2[] = "abd";
	char arr3[] = "abcdef";
	int ret1 = strcmp(arr1, arr2);
	int ret2 = strcmp(arr1, arr3);
	int ret3 = strcmp(arr2, arr3);
	printf("%d\n", ret1);
	printf("%d\n", ret2);
	printf("%d\n", ret3);
	return 0;
}

在这里插入图片描述
在这里插入图片描述
可以看到,标准里面的返回值是一个大于或小于0的数,但VS的编译器直接定为1和-1,其他编译器可能是返回一个大于0或小于0的数。

strcmp函数的模拟实现
#include<stdio.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++;
	}
	if (*str1 > *str2)
		return 1;
	else
		return -1;
}
int main()
{
	char arr1[] = "abcdef";
	char arr2[] = "abd";
	char arr3[] = "abcdef";
	int ret1 = my_strcmp(arr1, arr2);
	int ret2 = my_strcmp(arr1, arr3);
	int ret3 = my_strcmp(arr2, arr3);
	printf("%d\n", ret1);
	printf("%d\n", ret2);
	printf("%d\n", ret3);
	return 0;
}

在这里插入图片描述
模拟后发现结果是一样的。

  1. 还是和上面一样,因为函数中我们只是比较字符串,不需要进行修改,所以要对其进行const修饰,而且为了避免传入空指针,要对其进行断言处理。
  2. while循环中进行条件判断,相等就进入循环,并自增,不相等跳出循环,跳出循环后又分两种情况,一种是 *str1 > *str2,直接返回1,*str1 < *str2,直接返回-1。
  3. 相等的情况只可能出现在while循环中,因为不相等就直接跳出循环了,一旦当 *str1 == \0,也就说明 *str2也等于\0了,直接返回0就行了。

注意:这只是在模拟VS编译器里的结果,模拟其他编译器上的结果更加简单,不相等时直接返回他们对应的ASCII码的差值就行了。

#include<stdio.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[] = "abcdef";
	char arr2[] = "abs";
	char arr3[] = "abcdef";
	int ret1 = my_strcmp(arr1, arr2);
	int ret2 = my_strcmp(arr1, arr3);
	int ret3 = my_strcmp(arr2, arr3);
	printf("%d\n", ret1);
	printf("%d\n", ret2);
	printf("%d\n", ret3);
	return 0;
}

在这里插入图片描述

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

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

相关文章

5 个最佳 Mock 工具

在持续发展的前端开发领域&#xff0c;一套高效的自动化工具是关键。这篇文章将带你了解五个出色的模拟工具&#xff0c;它们能极大提升你的生产力、简化数据仿真&#xff0c;并提升接口测试效率。对于寻求提高工作流的前端开发者来说&#xff0c;它们是必不可少的。让我们开始…

牛客题霸-SQL入门篇(刷题记录二)

本文基于前段时间学习总结的 MySQL 相关的查询语法&#xff0c;在牛客网找了相应的 MySQL 题目进行练习&#xff0c;以便加强对于 MySQL 查询语法的理解和应用。 以下内容是牛客题霸-SQL入门篇剩余的第 21-39 道题目的 SQL 代码答案。 由于涉及到的数据库表较多&#xff0c;因…

Jira自动化的实用工具——ScriptRunner简介及最佳实践

近日&#xff0c;龙智举办的DevSecOps研讨会年终专场“趋势展望与实战探讨&#xff1a;如何打好DevOps基础、赋能创新”在上海圆满落幕。龙智Atlassian技术与顾问咨询团队&#xff0c;以及清晖、JamaSoftware、CloudBees等生态伙伴的嘉宾发表了主题演讲&#xff0c;分享他们在D…

【鸿蒙HarmonyOS开发笔记】常用组件介绍篇 —— Progress进度条组件

概述 Progress为进度条组件&#xff0c;用于显示各种进度。 参数 Progress组件的参数定义如下 Progress(options: {value: number, total?: number, type?: ProgressType})● value value属性用于设置当前进度值。 ● total total属性用于设置总值。 ● type type属…

小程序搜索排名优化二三事

小程序的优化主要是排名优化和性能优化两个版块。性能优化这方面主要靠开发者自己完善&#xff0c;我们团队提供的服务就是把产品的排名打上去&#xff0c;获得更多的自然流量&#xff0c;实现盈利。 如何提升小程序的搜索排名主要从如下几个方面出发&#xff1a; 首先要知道…

nodeJs 学习

常用快捷键 二、fs模块 回调函数为空&#xff0c;则表示写入成功&#xff01; 练习 const fs require(fs); fs.readFile(../files/成绩.txt, utf-8, (err, dataStr) > {if (err) {console.log(读取失败);return err;}console.log(读取成功);const arr dataStr.split( )co…

B007-springcloud alibaba 消息驱动 Rocketmq

目录 MQ简介什么是MQMQ的应用场景异步解耦流量削峰 常见的MQ产品 RocketMQ入门RocketMQ环境搭建环境准备安装RocketMQ启动RocketMQ测试RocketMQ关闭RocketMQ RocketMQ的架构及概念RocketMQ控制台安装 消息发送和接收演示发送消息接收消息 案例订单微服务发送消息用户微服务订阅…

华中师范大学伍伦贡联合研究院与南京大学联合主办——第三届人工智能、物联网和云计算技术国际会议(AIoTC 2024)

#先投稿&#xff0c;先送审# 第三届人工智能、物联网和云计算技术国际会议&#xff08;AIoTC 2024&#xff09; 2024 3rd International Conference on Artificial Intelligence, Internet of Things and Cloud Computing Technology 2024年8月2-4日 | 中国南京 重要信息 大…

1.详细解释单链表中的头结点;2.Java算法——力扣707题:设计链表

1.详细解释单链表中的头结点 在做这道算法之前&#xff0c;首先务必要弄明白三个问题&#xff1a; 对于含头节点的单链表&#xff0c; 头结点是否是第一个节点&#xff1f;单链表的长度是否包含该头节点&#xff1f;头结点是否有索引&#xff1f;如果有的话&#xff0c;又是多…

[保姆级教程]Windows安装MongoDB教程

文章目录 MongoDB安装包下载1.点击进入mongodb官网2.点击MongoDB Community Edition&#xff08;社区版&#xff09;&#xff0c;进入下图界面3.选择版本4.下载5.安装6.勾选同意协议&#xff0c;点击“Next"7.选择自定义安装8.点击“Next"9.修改到合适的地址10.点击i…

linux 使用docker安装 postgres 教程,踩坑实践

linux 使用docker安装 postgres 教程 踩坑实践,安装好了不能远程访问。 防火墙已关闭、postgres 配置了允许所有ip 访问、网络是通的。端口也是开放的&#xff0c;就是不能用数据库链接工具访问。 最后发现是云服务器端口没开 ,将其打开 到这一步完全正确了&#xff0c;但是…

Springboot笔记-01

简化spring应用开发&#xff0c;约定大于配置 简化Spring应用开发的一个框架&#xff1b; 整个Spring技术栈的一个大整合&#xff1b; J2EE开发的一站式解决方案&#xff1b; 优点&#xff1a; 快速创建独立运行的spring项目以及于主流框架集成 使用嵌入式的Servlet容器&#x…

HarmonyOS NEXT应用开发—多层级轮播图案例

介绍 本示例介绍使用ArkUIstack 组件实现多层级瀑布流。该场景多用于购物、资讯类应用。 效果图预览 使用说明 加载完成后显示轮播图可以左右滑动。 实现思路 SwiperDataSource&#xff0c;实现IDataSource接口的对象&#xff0c;用于LazyForEach加载数据。源码参考Swiper…

iframe重新加载最简单的方法!初次渲染和更新后渲染不会引起其他页面变动

我们项目有一个【加载场景】【切换场景】的需求&#xff0c;iframe是子组件&#xff0c;数据传过去之后没有重新加载 &#xff0c;我测试发现 重新渲染的时候&#xff0c;iframe的load方法没有重新加载&#xff0c;数据变动后也不会影响&#xff0c;然后我用了网上好几种方法&a…

一个简单的Vue2例子讲明白拖拽drag、移入dragover、放下drop的触发机制先后顺序

几个小细节说明&#xff1a; 执行顺序dragstart→dragover→drop被拖拽的物体必须要设置draggable"true"&#xff08;实际上只需要draggable就可以了&#xff0c;默认就是true&#xff09;&#xff0c;否者默认一般是不允许被拖拽的&#xff08;图片除外&#xff09;…

将main打包成jar;idea打包main为jar包运行

将main打包成jar&#xff1b;idea打包main为jar包运行 适用场景&#xff1a;可以封装一些小工具。 配置jar Maven中添加 <packaging>jar</packaging>将其打包为jar。 设置运行入口main 编译jar 看到jar输出 运行效果&#xff1a; 其中&#xff0c;三方依赖也被…

CLIP解读

1、引言 在计算机视觉领域&#xff0c;通常需要经过训练模型来实现对预定类别目标预测&#xff08;如分类、检测等任务&#xff09;&#xff0c;但是这种形式会限制模型的通用性。比如我们训练完了一个猫狗分类模型&#xff0c;如果现在希望识别一只老虎&#xff0c;那么原来训…

【Qt】Qt中的常用属性

需要云服务器等云产品来学习Linux可以移步/-->腾讯云<--/官网&#xff0c;轻量型云服务器低至112元/年&#xff0c;新用户首次下单享超低折扣。 目录 一、QWidget属性一览 二、属性enabled(可用状态) 三、属性geometry(修改位置和尺寸) 1、QRect类型的结构 2、geome…

nginx介绍及搭建

架构模型 Nginx是由一个master管理进程、多个worker进程组成的多进程模型。master负责管理worker进程&#xff0c;worker进程负责处理网络事件&#xff0c;整个框架被设计为一种依赖事件驱动、异步、非阻塞的模式。 优势&#xff1a; 1、充分利用多核&#xff0c;增强并发处理…

FMEA赋能人工智能:开启智能风险预防新纪元!

在数字化浪潮席卷全球的今天&#xff0c;人工智能&#xff08;AI&#xff09;已经渗透进我们生活的方方面面&#xff0c;而如何确保AI系统的稳定与安全&#xff0c;成为摆在科技界和工业界面前的一大挑战。FMEA&#xff08;失效模式与影响分析&#xff09;作为一种经典的质量管…