【C语言】高频考点,关于字符串函数的使用以及模拟实现

大家好呀!👋这个是付青云同学的博客,是一名大一在校生哦!😁😁
目前一直在学习C语言。🐸
写博客是为了来记录我的学习过程,同时也希望通过博客能够帮助到需要帮助的人。
如果我的博客可以帮助到你,不妨给我一个关注哦😁


写在前面

最近学完了库函数中的部分字符串函,在学习的过程中我发现利用这些库函数,我们可以非常方便的写程序;同时,了解这些库函数的实现原理,我们也可以更灵活地利用这些库函数并且也加深了我们对C语言的理解。
😎😎😎
那么现在,废话不多说,让我们开始吧!


求字符串长度的函数🤨

strlen函数

我想这个函数大家都很熟悉了吧
用于计算字符串的长度
但是有几个注意事项:

  • 字符串是将 ‘\0’ 作为结束标志,strlen函数返回的是在字符串中 ‘\0’ 前面出现的字符个数不包含 ‘\0’ )。
  • 参数指向的字符串必须要以’\n’结束
  • 注意函数的返回值为size_t,是无符号的(unsigned int)
int main()
{
	char arr[] = "abcdef";
	printf("%d\n", strlen(arr));//6
	return 0;
}

strlen函数的模拟实现

这个函数其实有很多种方法可以实现

一般实现
#include <assert.h>
int my_strlen(const char* str)
{
	int count = 0;
	assert(str);
	while (*str)
	{
		count++;
		str++;
	}
	return count;
}
递归实现
int my_strlen(char* str)
{
	if (*str == '\0')
	{
		return 0;
	}
	else
	{
		return  my_strlen(++str) + 1;//注意这里要先自加
	}
}
指针-指针实现

这里有个知识点:指针-指针=两指针之间的元素个数

int my_strlen(char* str)
{
	char* a = str;
	while (*str)
	{
		str++;
	}
	char* b = str;
	return b - a;
}
拓展

在这里,我把my_strlen的返回值为int类型,并没有像库函数那样设置为size_t的类型。
两种方法各有好处:

  • 使用size_t:考虑到字符串长度不可能为一个负值,所以使用无符号类型
  • 使用int:在进行字符串长度相减的时候可以有负数,以免出现离谱的数据

有这样有个题目:

#include <stdio.h>

int main()
{
	const char* str1 = "abcdef";
	const char* str2 = "abc";
	if (strlen(str2) - strlen(str1) > 0)
	{
		printf("str2>str1\n");
	}
	else
	{
		printf("srt1>str2\n");
	}
	return 0;
}

这个题目很多不熟悉strlen函数的人都会认为:str2的长度小于str1;事实上确实如此,但是,strlen函数返回的值是无符号的,所以两个相减得到的是一个无符号类型的值,因此肯定是大于0的。


长度不受限的字符串函数🤨

strcpy函数

拷贝函数
注意:

  • 源字符串必须以 ‘\0’ 结束。
  • 会将源字符串中的 ‘\0’ 拷贝到目标空间。
  • 目标空间必须足够大,以确保能存放源字符串。
  • 目标空间必须可变
char* str = "xxxxxxxxxxxxxx";//❌
#include <stdio.h>
#include <string.h>

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

strcpy的模拟实现

#include <stdio.h>
#include <assert.h>

char* my_strcpy(char* dest, const char* src)//二、const为修饰常变量
{//返回型
    assert(src != NULL);//一、断言

    char* ret=dest
    while (*dest++ = *src++)//2、当位置写反时,程序错误
	{//既copy了\0,又使得循环停止
		;
	}
    return ret;//返回目标空间的起始地址
}
int main()
{
	char arr1[20] = "*******************";
	char arr2[] = "hello!";
    //使用strcpy时arr2中的\0也会被拷贝
	my_strcpy(arr1, arr2);//1、当arr2为NULL时,无法进行解引用操作
     //链式访问
	printf("%s\n", arr1);
	return 0;
}

strcat函数

追加函数
注意事项:

  • 源字符串必须以 ‘\0’ 结束。
  • 目标空间必须有足够的大,能容纳下源字符串的内容。
  • 目标空间必须可修改
  • 不能追加自己

模拟实现strcat函数

#include <stdio.h>
#inlcude <assert.h>

char* my_strcat(char* str1, const char* str2)
{
	char* ret = str1;
	assert(str1 && str2);
	//1、找到目标字符串的\0
	while (*str1)
	{
		str1++;
	}
	//2、追加源字符串,包含\0
	while (*str1++ = *str2++)
	{
		;
	}
	return ret;//返回目标空间的起始地址
}

int main()
{
	char str1[20] = "abc";
	char str2[] = "def";
	printf("%s\n", my_strcat(str1, str2));//abcdef
	return 0;

strcmp函数

比较函数
规则:

  • 第一个字符串大于第二个字符串,则返回大于0的数字
  • 第一个字符串等于第二个字符串,则返回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 = "abcd";
	char* arr2 = "abdce";
	//char arr1[]="abc";
 	//char arr2[]="abc";
	int n = my_strcmp(arr1, arr2);
	printf("%d\n", n);
	return 0;
}

长度受限的字符串函数🤨

关于这些函数,用法跟长度不受限的函数的用法差不多,只不过在后面多了一个限制长度的参数,具体用法可以查阅C语言文档

strncpy

注意:

  • 如果源字符串的长度小于num,则拷贝完源字符串之后,在目标的后边追加0,直到num个。

strncat

strncmp


字符串查找🤨

strstr函数

在一个字符串中找另外一个字符串(包含与被包含)

模拟实现strstr函数

#include <stdio.h>
#include <assert.h>

char* my_strstr(const char* str1, const char* str2)
{
	assert(str1 && str2);

	char* s1 = NULL;
	char* s2 = NULL;
	const char* sp = str1;

	if (*str2 == '\0')
	{
		return (char*)str1;
	}

	while (*sp)
	{
		s1 = sp;
		s2 = str2;
  //*s1 && *s2的目的是防止“abcde”和“bcde”最后都='\0'时内存溢出
		while (*s1 && *s2 && (*s1 == *s2))
		{
			s1++;
			s2++;
		}
		if (*s2 == '\0')
		{
			return (char*)sp;
		}
		sp++;
	}
	return NULL;
}

int main()
{
	char arr1[] = "abccccdefgh";
	char arr2[] = "ccdef";
	printf("%s\n", my_strstr(arr1, arr2));
	return 0;
}

strtok函数

切割字符串

  • strtok函数找到str中的下一个标记,并将其用 \0 结尾,返回一个指向这个标记的指针
    (注:strtok函数会改变被操作的字符串,所以在使用strtok函数切分的字符串一般都是临时拷贝的内容并且可修改。)
  • strtok函数的第一个参数不为 NULL ,函数将找到str中第一个标记,strtok函数将保存它在字符串中的位置
  • strtok函数的第一个参数为 NULL ,函数将在同一个字符串中被保存的位置开始查找下一个标记
  • 如果字符串中不存在更多的标记,则返回 NULL 指针。
#include <stdio.h>
#include <string.h>

int main()
{
	char arr1[] = "kenobi@qq.com";
	char *arr2 = "@.";
	char arr[20] = { 0 };
	char* str = NULL;
	strcpy(arr, arr1);
	for (str = strtok(arr, arr2); str != NULL; str = strtok(NULL, arr2))
	{
		printf("%s\n", str);
	}
	return 0;
}

错误信息报告🤨

strerror函数

返回错误码信息

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

int main()
{
	FILE* pf = fopen("test.c", "r");//实际上没有这个test.c文件

	if (pf == NULL)
	{
		printf("%s\n", strerror(errno));
		return 1;
	}
 //关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

perror函数

这个函数的使用比strerror函数更加方便
可以直接打印错误码信息

#include <stdio.h>

int main()
{
	FILE* pf = fopen("test.c", "r");//实际上没有这个test.c文件

	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
 //关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

结束🥳🥳🥳

创作不易
你的关注与支持就是我创作的动力
如果这篇文章对你有帮助的话,不妨给个三连。
非常感谢!!
😁😁😁

  • 63
    点赞
  • 49
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 44
    评论
评论 44
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

付青云同学

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值