字符串_字符函数和字符串函数

C语言中对字符和字符串的处理很是频繁,但是C语言本身是没有字符串类型的,字符串通常放在常量字符串中或者字符数组中。

字符串常量适用于那些对它不做修改的字符串函数。

目录

1.函数介绍

1.1strlen

1.1.1strlen函数的模拟实现

1.2strcpy

1.2.1strcpy函数的模拟实现

1.3 strcat(字符串追加)

1.3.1strcat函数的模拟实现

1.4strcmp

1.4.1strcmp函数的模拟实现

1.5strncpy、strncat、strncmp的使用 

1.5.1strncat函数的模拟实现 

1.5.2strcpy函数的模拟实现 

1.5.3strcmp函数的模拟实现 

1.6strstr(查找子串的一个函数)

1.6.1strstr函数的模拟实现

1.7strtok(切割字符串)

1.8 strerror

1.函数介绍

1.1strlen

size_t strlen(const char* str);

注意:

字符串以'\0'作为结束标志,strlen函数返回的是在字符串中'\0'前面出现的字符个数但不包括'\0'。 

参数指向的字符串必须要以'\0'结束。

函数的返回值为size_t,是无符号的。(size_t转为定义即unsigned int)

1.1.1strlen函数的模拟实现

#include<stdio.h>
//法一:计数器实现
int my_strlen(char* str)
{
	int count = 0;
	while (*str != '\0')
	{
		str++;
		count++;
	}
	return count;
}
//法二:递归实现
int my_strlen(char* str)
{
	if (*str)
		return 1 + my_strlen(str + 1);
	else
		return 0;
}
//法三:指针-指针
int my_strlen(char* str)
{
	char* p = str;
	while (*str)
	{
		str++;
	}
	return str - p;
}
int main()
{
	char arr[] = "abcdef";
	int len = my_strlen(arr);
	printf("%d\n", len);
	return 0;
}

1.2strcpy

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

注意:

源字符串必须以'\0'结束。

会将源字符串中的'\0'拷贝到目标空间。

目标空间必须足够大,以确保能存放源字符串。

目标空间必须可变

1.2.1strcpy函数的模拟实现

#include<stdio.h>
char* my_strcpy(char* dest, char* stc)
{
	char* ret = dest;
	while (*stc!='\0')
	{
		*dest++ = *stc++;
	}
	*dest = *stc;
	return ret;
}
//实际上strcpy返回的是目标空间的起始地址,因为我们是把源字符串拷贝到目标空间里,目标空间发生变化
int main()
{
	char arr1[20] = { 0 };
	char arr2[] = "hello world";
	my_strcpy(arr1, arr2);
	printf("%s\n", arr1);
	return 0;
}

1.3 strcat(字符串追加)

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

注意:

源字符串必须以'\0'结束。

目标空间必须足够大,以确保能存放源字符串。

目标空间必须可修改。

1.3.1strcat函数的模拟实现

#include<stdio.h>
char* my_strcat(char* dest, char* stc)
{
	char* ret = dest;
    //1.找到目标空间末尾的\0
	while (*dest != '\0')
	{
		dest++;
	}
    //2.拷贝字符串
	while (*dest++ = *stc++)//相当于strcpy函数
	{
		;
	}
	return ret;
}
int main()
{
	char arr[20] = "hello ";
	my_strcat(arr, "world");
	printf("%s\n", arr);
	return 0;
}

 尽量不要自己给自己追加,可能会导致程序崩溃,因为在找到末尾的\0时会覆盖掉,导致自己破坏了自己的内容,从而陷入死循环。

1.4strcmp

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

#include<stdio.h>
#include<string.h>
int main()
{
	char arr1[] = "zhangsan";
	char arr2[] = "zhangsan";
	int ret = strcmp(arr1, arr2);
	if (ret < 0)
		printf("<\n");
	else if (ret == 0)
		printf("==\n");
	else
		printf(">\n");
	return 0;
}

标准规定:第一个字符串大于第二个字符串,则返回大于0的数字;第一个字符串等于第二个字符串,则返回0;第一个字符串小于第二个字符串,则返回小于0的数字。

1.4.1strcmp函数的模拟实现

#include<stdio.h>
#include<string.h>
int my_strcmp(char* str1, char* str2)
{
	while (*str1 == *str2)
	{
		if (*str1 == '\0')
			return 0;
		str1++;
		str2++;
	}
	return (*str1 - *str2);
}
int main()
{
	char arr1[] = "zhangsan";
	char arr2[] = "zhangsand";
	int ret = my_strcmp(arr1, arr2);
	if (ret < 0)
		printf("<\n");
	else if (ret == 0)
		printf("==\n");
	else
		printf(">\n");
	return 0;
}

1.5strncpy、strncat、strncmp的使用 

上面几个字符串函数不受长度限制,会有安全隐患。这里提供几个长度受限制的字符串函数:strncpy、strncat、strncmp,这三个函数和strcpy、strcat、strcmp使用方法基本一致,但其多了一个n,并且在调用函数时多了一个参数,其用来限制具体操作的字符个数。

cahr* strncat(char* destination,char* source,size_t num) ;

char* strcpy(char* destination,char* source,size_t num) ;

char* strcmp(char* destination,char* source,size_t num) ;

我们拿strncat举例: 

#include<stdio.h>
int main()
{
	char arr1[20] = "abcdef";
	char arr2[] = "hello";
	strncat(arr1, arr2,3);
	printf("%s\n", arr1);
	return 0;
}

 1、将source指向字符串的前num个字符追加到destination指向的字符串末尾,再追加一个\0字符。

2、如果source指向的字符串的长度小于num的时候,只会将字符串中到\0的内容追加destination指向的的字符串末尾。

1.5.1strncat函数的模拟实现 

#include<stdio.h>
char* my_strncat(char* dest, char* stc, int num)
{
	char* p = dest;
	while (*dest)
	{
		dest++;
	}
	while (*stc&&num--)
	{
		*dest++ = *stc++;
	}
	*dest = '\0';
	return p;
}
int main()
{
	char arr1[20] = "abcdef";
	char arr2[] = "hello";
	my_strncat(arr1, arr2,3);
	printf("%s\n", arr1);
	return 0;
}

1.5.2strcpy函数的模拟实现 

#include<stdio.h>
char* my_strncpy(char* dest, char* stc, int num)
{
	char* p = dest;
	while (num--)
	{
		*dest++ = *stc++;
	}
	return p;
}
int main()
{
	char arr1[20] = "abcdef";
	char arr2[] = "hello";
	my_strncpy(arr1, arr2,3);
	printf("%s\n", arr1);
	return 0;
}

1.5.3strcmp函数的模拟实现 

 

#include<stdio.h>
int my_strncmp(char* dest, char* stc, int num)
{
	while (num && dest && stc)
	{
		if (*dest > *stc)
		{
			return 1;
		}
		if (*dest < *stc)
		{
			return -1;
		}
		num--;
		dest++;
		stc++;
	}
	return 0;
}
int main()
{
	char arr1[20] = "abcdef";
	char arr2[20] = "abcdeq";
	int ret=my_strncmp(arr1, arr2, 6);
	if (ret>0)
		printf(">\n");
	else if (ret<0)
		printf("<\n");
	else
		printf("==\n");
	return 0;
}

1.6strstr(查找子串的一个函数)

char* strstr(const char* str1,const char* str2);

例子:

#include<stdio.h>

int main()
{
    char arr1[] = "abcdef";
    char arr2[] = "bcde";
    char* ret = strstr(arr1, arr2);//如果能找到,返回的是b的地址,找不到返回的是空指针
    if (ret == NULL)
        printf("找不到\n");
    else
        printf("%s\n",ret);
    return 0;
}

1.6.1strstr函数的模拟实现

#include<stdio.h>
char* my_strstr(const char* str1,const char* str2)//尽量不要让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 p;
		p++;
	}
	return NULL;
}
int main()
{
	char arr1[] = "abbbcdef";
	char arr2[] = "bcde";
	char* ret = my_strstr(arr1, arr2);//如果能找到,返回的是b的地址,找不到返回的是空指针
	if (ret == NULL)
		printf("找不到\n");
	else
		printf("%s\n",ret);
	return 0;
}

1.7strtok(切割字符串)

char* strtok(char* str,char* sep);

sep参数是一个字符串,定义了用作分隔符的字符集合;

第一个参数指定一个字符串,它包含了0个或多个由sep字符串中一个或者多个分隔符的标记;

strtok函数找到str中的下一个标记,并将用\0结尾,返回一个指向这个标记的指针;

strtok函数的第一个参数不为NULL,函数将找到str中第一个标记,strtok函数将保存它在字符串中的位置;

strtok函数的第一个参数为NULL,函数将在同一个字符串中被保存的位置开始,查找下一个标记;

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

#include<stdio.h>
int main()
{
	const char* sep = "@.";
	char data[] = "http@csdn.com";
	char cp[30] = { 0 };//因为strtok函数在找字符串时会把标记符改成\0,所以可以拷贝一个临时数组
	strcpy(cp, data);
	char* ret = strtok(cp, sep);
	printf("%s\n", ret);
	ret = strtok(NULL, sep);//这个空指针代表从保存好的位置不断向后找,是一种状态。
	printf("%s\n", ret);
	ret = strtok(NULL, sep);
	printf("%s\n", ret);
    //这样写是比较挫的,换种写法
    char* ret = NULL;
    for (ret = strtok(cp, sep); ret != NULL; ret = strtok(NULL, sep))
    {
	    printf("%s\n", ret);
    }
	return 0;
}

1.8 strerror

char* strerror(int errnum);

返回错误码,所对应的错误信息。

举个例子:

#include<stdio.h>
#include<errno.h>
#include<string.h>
int main()
{
	//errno-C语言设置的一个全局的错误码存放的变量
	FILE* pf = fopen("text.txt", "r");
	if (pf == NULL)
	{
		printf("%s\n", strerror(errno));
		return 1;
	}
	return 0;
}

字符分类函数:

函数如果它的参数符合下列条件就返回真
iscntrl任何控制字符
isspace空白字符:空格' ',换页'\f',换行'\n',回车'\r',制表符'\t'或者垂直制表符'\v'
isgigit十进制数字0~9
isxdigit十六进制数字,包括所有十进制数字,小写字母a~f,大写字母A~F
islower小写字母a~z
isupper大写字母A~Z
isalpha字母a~z或A~Z
isalnum字母或者数字,a~z,A~Z,0~9
ispunct标点符号,任何不属于数字或者字母的图形字符(可打印)
isgraph任何图形字符
isprint任何可打印字符,包括图形字符和空白字符

 详情可参考cpulscpuls.com网站,上面有详细介绍

字符转换:

int tolower(int c);//转小写

int toupper(int c);//转大写

 

 

 

 

 

 

 

 

 

 

 

 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

性能测试学习二

瓶颈的精准判断 TPS曲线 tps图 响应时间图 拐点在哪里呢? 这是一个阶梯式增加的场景,拐点在第二个压力阶梯上就出现了,因为响应时间增加了,tps增加的却不多,在第三个阶段时,tps增加的就更少了,响应时间也在不断增加,所以性能瓶颈在加剧,越往后越明显【tps的增长,…

【35分钟掌握金融风控策略29】贷中模型调额调价策略

目录 贷中客户风险管理和客户运营体系 用信审批策略 用信审批策略决策流与策略类型 贷中预警策略 对存量客户进行风险评级 基于客户的风险评级为客户匹配相应的风险缓释措施和建议 调额策略 基于定额策略的调额策略 基于客户在贷中的风险表现的调额策略 调价策略 存…

鸿蒙开发接口Ability框架:【ApplicationContext】

ApplicationContext ApplicationContext模块提供开发者应用级别的的上下文的能力&#xff0c;包括提供注册及取消注册应用内组件生命周期的监听接口。 说明&#xff1a; 开发前请熟悉鸿蒙开发指导文档&#xff1a; gitee.com/li-shizhen-skin/harmony-os/blob/master/README.m…

留学资讯 | 2024英国学生签证申请需要满足哪些条件?

英国移民局于2020年9月10日发布了《移民规则变更声明: HC 707》&#xff0c;对学生签证制度进行了全面改革。该法案于2020年10月5日正式生效。根据此法案&#xff0c;新的学生签证——The Student and Child Student Routes学生和儿童学生路线&#xff0c;将替代原先的Tier 4学…

基于java的超级玛丽游戏的设计与实现(论文 + 源码)

Java的超级玛丽游戏.zip资源-CSDN文库https://download.csdn.net/download/JW_559/89313347 基于java的超级玛丽游戏的设计与实现 摘要 近年来&#xff0c;Java作为一种新的编程语言&#xff0c;以其简单性、可移植性和平台无关性等优点&#xff0c;得到了广泛地应用。J2SE称…

链接表存储图(C++注释详解): 构建表 深度优先遍历 (DFS)

链接表的结构体单元: #define size 100 typedef struct node {int idx;//下一个节点的索引int wt;//权重, 也可根据实际情景存储边的信息struct node* next; }Node; Node* hd[size]; // 存储图的邻接表 链接表的的构建: int main() {int n, m;cin >> n >> m; //…

SOLIDWORKS 2024零件特征功能增强

如大家所知&#xff0c;达索系统SOLIDWORKS每年都会发布新版本以主动响应客户的需求。现有客户使用的版本并不一样&#xff0c;所以在文档数据交流方面存在一定困难。同时工厂中的其它部门都会与产品研发部门进行协作&#xff0c;所以我们需要更强大的软件功能快速接收和处理模…

AnyMP4 Video Converter for Mac/Win - 视频转换的卓越之选

在当今数字化的时代&#xff0c;视频内容无处不在&#xff0c;而拥有一款强大的视频转换器就显得至关重要。AnyMP4 Video Converter for Mac/win 正是这样一款出类拔萃的工具&#xff0c;为您带来高效、便捷的视频转换体验。 这款视频转换器具备令人惊叹的功能。它支持广泛的视…

Shell之(数组)

目录 一、shell数组 1.数组的定义 2.定义数组的方法 第一种 第二种 第三种 第四种 3.数组分片 4. 数组字符替换 临时替换 永久替换 5.删除数组 删除指定的下标 删除整组 6.数组遍历和重新定义 7.数组追加元素 方式一&#xff1a;指定位置添加 方法二&a…

React: memo

React.memo 允许你的组件在 props 没有改变的情况下跳过重新渲染。 const MemoizedComponent memo(SomeComponent, arePropsEqual?)React 通常在其父组件重新渲染时重新渲染一个组件。你可以使用 memo 创建一个组件&#xff0c;当它的父组件重新渲染时&#xff0c;只要它的新…

阅读一些精华(老文献)

本文设置的初衷是诺贝尔化学奖得主Sir Fraser Stoddart说过&#xff1a;我们不能局限于最近几年的工作&#xff0c;而要往几十年前看&#xff0c;说不定因为之前的一些技术原因&#xff0c;导致当初的方法没有实现&#xff0c;现在可以实现了。 1.Variable-Rate Transmission f…

垃圾分类管理系统java项目

文章目录 垃圾分类管理系统一、项目演示二、项目介绍三、系统部分功能截图四、部分代码展示五、底部获取项目&#xff08;9.9&#xffe5;带走&#xff09; 垃圾分类管理系统 一、项目演示 垃圾分类管理系统 二、项目介绍 系统角色&#xff1a;管理员、用户 1、登录、注册功能…

服务器之间实现免密码传输文件(scp免密传输)

问题&#xff1a;需要定时将本服务器的文件传输到指定服务器上作为备份 通过scp实现不同服务器之间的文件传输 正常使用scp传输文件 传输文件命令&#xff1a;scp /data/文件 root服务器地址&#xff1a;/指定目录 传输文件夹命令&#xff1a;scp -r /data/文件 root服务…

Unity与Andriod的交互

Unity与安卓的信息交互 这次分享的不同于传统的方式AndroidJavaClass("com.unity3d.player.UnityPlayer") 如果是新手的话&#xff0c;请看 交互新手教程 这里讲的是在Unity中调用java代码&#xff0c;或者在unity中传参到java中&#xff0c;在Java代码中运行。 以下…

css画三角形

使用border div {border-top: 50px solid yellowgreen;border-bottom: 50px solid deeppink;border-left: 50px solid bisque;border-right: 50px solid chocolate; }如果想要单个的三角形&#xff0c;把其它三边的颜色设为transparent即可 使用 conic-gradient 绘制三角形 …

ADS FEM 仿真设置

1、EM Simulator 选择FEM。 2、在layout界面打开的EM功能&#xff0c;这里不需要操作。 3、Partitioning 不需要操作。 4、没有叠层的话需要新建&#xff0c;过孔可以在叠层处右键添加。 5、端口需要设置GND layer。 6、设置仿真频率。 7、Output plan。 8、Options 设置 介质…

低空经济:无人机竞赛详解

无人机竞赛市场近年来呈现出蓬勃发展的态势&#xff0c;其市场价值不仅体现在竞赛本身&#xff0c;还体现在推动无人机技术创新、拓展应用场景以及促进产业链发展等多个方面。 一、比赛项目介绍 无人机竞赛通常分为多个项目&#xff0c;包括竞速赛、技巧赛、航拍赛等。每个项目…

LeetCode494:目标和

题目描述 给你一个非负整数数组 nums 和一个整数 target 。 向数组中的每个整数前添加 ‘’ 或 ‘-’ &#xff0c;然后串联起所有整数&#xff0c;可以构造一个 表达式 &#xff1a; 例如&#xff0c;nums [2, 1] &#xff0c;可以在 2 之前添加 ‘’ &#xff0c;在 1 之…

GPT-4o:全面深入了解 OpenAI 的 GPT-4o

GPT-4o&#xff1a;全面深入了解 OpenAI 的 GPT-4o 关于 GPT-4o 的所有信息ChatGPT 增强的用户体验改进的多语言和音频功能GPT-4o 优于 Whisper-v3M3Exam 基准测试中的表现 GPT-4o 的起源追踪语言模型的演变GPT 谱系&#xff1a;人工智能语言的开拓者多模式飞跃&#xff1a;超越…

优秀博士学位论文分享:复杂场景下高精度有向目标检测的研究

优秀博士学位论文代表了各学科领域博士研究生研究成果的最高水平&#xff0c;本公众号近期将推出“优秀博士学位论文分享”系列文章&#xff0c;对人工智能领域2023年优秀博士学位论文进行介绍和分享&#xff0c;方便广大读者了解人工智能领域最前沿的研究进展。 “博士学位论…