C语言:字符函数和字符串函数

在编程的过程中,我们经常要处理字符和字符串,为了方便操作字符和字符串,C语言标准库中提供了一系列库函数,接下来我们就学习一下这些函数。

目录

  • 1. 字符分类函数
  • 2.字符转换函数
  • 3. strlen的使用和模拟实现
  • 4. strcpy的使用和模拟实现
  • 5. strcat的使用和模拟实现
  • 6. strcmp的使用和模拟实现
  • 7. strncpy函数的使用
  • 8. strncat函数的使用
  • 9. strncmp函数的使用
  • 10. strstr的使用和模拟实现
  • 11. strtok函数的使用
  • 12.strerror函数的使用

1. 字符分类函数

C语言中有一系列的函数是专门做字符分类的,也就是一个字符是属于什么类型的字符的。
这些函数的使用都需要包含一个头⽂件是 ctype.h

函数 如果它的参数符合下列条件就返回真
iscntrl 任何控制字符
isspace 空白字符:空格‘ ’,换页 ‘\f’,换行 ‘\n’,回车 ‘\r’,制表符‘\t’或者垂直制表符‘\v’
isdigit 十进制数字 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 任何可打印字符,包括图形字符和空白字符

这些函数的使用方法非常类似,我们就讲解⼀个函数的例子,其他的非常类似:

int islower ( int c );
islower 是能够判断参数部分的 c 是否是小写字母的。
通过返回值来说明是否是小写字母,如果是小写字母就返回非0的整数,如果不是小写字母,则返回0。
练习:
写一个代码,将字符串中的小写字母转大写,其他字符不变。

#include <stdio.h>
#include <ctype.h>
int main ()
{
	int i = 0;
	char str[] = "Test String.\n";
	char c;
	while (str[i])
	{
		c = str[i];
		if (islower(c))//判断是否是小写字母
		c -= 32;
		putchar(c);//输出一个字符
		i++;
	}
	return 0;
}

2.字符转换函数

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

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

上面的代码,我们将小写转大写,是-32完成的效果,有了转换函数,就可以直接使用 toupper 函数。

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

3. strlen的使用和模拟实现

库中strlen的参数和返回类型:
size_t strlen ( const char * str );

  • 字符串以 '\0' 作为结束标志,strlen函数返回的是在字符串中 '\0' 前面出现的字符个数(不包含 '\0' )。
  • 参数指向的字符串必须要以 '\0' 结束。
  • 注意函数的返回值为size_t,是⽆符号的( 易错
  • strlen的使用需要包含头文件

strlen的模拟实现:
方式1:

//计数器⽅式
int my_strlen(const char * str)
{
	int count = 0;
	assert(str);//断言指针不为空指针
	while(*str)//当解引用访问的字符不为‘\0’的时候进入循环
	{
		count++;
		str++;//指针指向下一个字符
	}
	return count;
}

方式2:

//不能创建临时变量计数器(使用递归的方式)
int my_strlen(const char * str)
{
	assert(str);//断言指针不为空指针
	if(*str == '\0')//递归的终止条件
	return 0;
	else
	return 1+my_strlen(str+1);//每次调用函数都会越来越接近终止条件
}

方式3:

//指针-指针的⽅式
int my_strlen(char *s)
{
	assert(str);
	char *p = s;
	while(*p != ‘\0’ )
		p++;
	return p-s;//指针与指针相减得到的是指针之间的元素个数
}

4. strcpy的使用和模拟实现

库中strcpy的参数和返回类型:
char* strcpy(char * destination, const char * source )

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

strcpy的模拟实现:

//dest为目标空间的起始地址,src为源字符串起始地址
char *my_strcpy(char *dest, const char*src)
{
	char *ret = dest;
	assert(dest != NULL);//断言指针不为空指针
	assert(src != NULL);
	while((*dest++ = *src++))
	//由于是后置++,所以先使用没+1之前的值进行解引用和赋值操作
	//最后会把'\0'的值也赋过去,虽然表达式结果为假,但是这个过程已经执行了
	{
		;//当不要进行任何操作的时候,放一个空语句就可以
	}
	return ret;
}

5. strcat的使用和模拟实现

  • 源字符串必须以 '\0' 结束。
  • 目标字符串中也得有 \0 ,否则没办法知道从哪⾥开始追加。
  • 目标空间必须有足够的大,能容纳下源字符串的内容。
  • 目标空间必须可修改。

模拟实现strcat函数:

char *my_strcat(char *dest, const char*src)
{
	char *ret = dest;
	assert(dest != NULL);//断言指针不为空指针
	assert(src != NULL);
	while(*dest)//先让dest指针指向'\0'
	{
		dest++;
	}
	while((*dest++ = *src++))//追加
	{
		;
	}
	return ret;
}

6. strcmp的使用和模拟实现

  • 第一个字符串大于第⼆个字符串,则返回大于0的数字
  • 第一个字符串等于第⼆个字符串,则返回0
  • 第一个字符串小于第⼆个字符串,则返回小于0的数字
  • 那么如何判断两个字符串? 比较两个字符串中对应位置上字符ASCII码值的大小。

strcmp函数的模拟实现:

int my_strcmp (const char * str1, const char * str2)
{
	int ret = 0 ;
	assert(src != NULL);//断言指针不为空指针
	assert(dest != NULL);
	while(*str1 == *str2)
	{
		if(*str1 == '\0')//两字符串相等的情况下,如果不加这条语句,则指针会一直++,就会越界
		return 0;
		str1++;
		str2++;
	}
	return *str1-*str2;
}

7. strncpy函数的使用

库中strncpy的参数和返回类型:
char * strncpy ( char * destination, const char * source, size_t num )

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

strncpy的模拟实现:

char* My_strncpy(char* dest,const char* src, int num)
{
	assert(dest && src);
	char* ret = dest;
	while (num--)
	{
		*dest = *src;
		dest++;
		if (*src=='\0')//当源字符串遍历到'\0'的时候,src指针就不需要移动了
			continue;
		src++;
	}
	return ret;
}
int main()
{
	char str1[20] = "xxxxxxxxxxxxxx";
	char str2[] = "666666";
	My_strncpy(str1, str2, 10);
	printf(str1);
	return 0;
}


在调试起来之后我们可以看到,源字符串(str2)长度为6,小于10,所以拷贝完源字符串之后,又在目标字符串追加了0,直到10个。

8. strncat函数的使用

库中strncat的参数和返回类型:
char * strncat ( char * destination, const char * source, size_t num )

  • 将source指向字符串的前num个字符追加到destination指向的字符串末尾,再追加⼀个 \0 字符
  • 如果source指向的字符串的长度小于num的时候,只会将字符串中的 \0 的内容追加到destination指向的字符串末尾
/* strncat example */
#include <stdio.h>
#include <string.h>
int main ()
{
	char str1[20];
	char str2[20];
	strcpy (str1,"To be ");
	strcpy (str2,"or not to be");
	strncat (str1, str2, 6);
	printf("%s\n", str1);
	return 0;
}

9. strncmp函数的使用

库中strncmp的参数和返回类型:
int strncmp ( const char * str1, const char * str2, size_t num )
比较str1和str2的前num个字符,如果相等就继续往后比较,最多比较num个字母,如果num个字符都相等,就是相等返回0,如果提前发现不一样,就提前结束,判断两指针指向的字符大小,大的字符所在的字符串大于另外一个。
strncmp的模拟实现:

int My_strncmp(const char* a1, const char* a2,int num)
{
	assert(a1 && a2);//断言指针变量a1,a2,以防指针为空
	while (num--)//比较的字符个数
	{
		while (*a1 == *a2)
		{
			if (*a1 == '\0')//a1字符串解引用为'\0'的时候,a2也必为'\0',说明两字符串相等
				return 0;
			a1++;
			a2++;
		}
		return *a1 - *a2;//走到这里就说明两指针指向的字符不想等,那直接返回差就可以
	}
}
int main()
{
	char arr1[] = "erd";
	char arr2[] = "erdfvj";
	if (My_strncmp(arr1, arr2,4) > 0)
		printf("arr1>arr2");
	else if (My_strncmp(arr1, arr2,4) < 0)
		printf("arr1<arr2");
	else
		printf("arr1=arr2");
	return 0;
}

10. strstr的使用和模拟实现

库中strstr的参数和返回类型:
char * strstr ( const char * str1, const char * str2)

  • 函数返回字符串str2在字符串str1中第一次出现的位置
  • 字符串的比较匹配不包含 \0 字符,以 \0 作为结束标志
#include <stdio.h>
#include <string.h>
int main()
{
	char str[] = "This is a simple string";
	char* pch=strstr(str, "a simple");
	printf("%s\n", pch);//打印的就是:a simple string
	return 0;
}

strstr的模拟实现:

char * strstr (const char * str1, const char * str2)
{
	char *cp = (char *) str1;//用来记录能够与str2中字符相匹配的地址
	char *s1, *s2;//创建两指针变量分别遍历str1字符串和str2字符串,比较字符是否相等
	if ( *str2=='\0' )
	return((char *)str1);
	while (*cp)//当指针cp解引用之后不为'\0'时,进入循环
	{
		s1 = cp;
		s2 = (char *) str2;//每次比较完不相等之后,s2就要回归到字符串
		的起始地址,这也是为什么要创建s2指针的原因,str2要记录字符串的起始地址
		while ( *s1==*s2 && *s1 !='\0' && *s2 !='\0' )
		{
		s1++;
		s2++;
		}
		//当*s2='\0'时说明已经在str1字符串中找到了str2字符串
		//那直接返回str1与str2相匹配的起始地址
		if (*s2=='\0')
		return cp;
		cp++;
	}
	//走到这里,就说明已经遍历完str1字符串,还是没有找到str2字符串
	return(NULL);
}

11. strtok函数的使用

库中strtok的参数和返回类型:
char * strtok ( char * str, const char * sep)

  • sep参数指向一个字符串,定义了用作分隔符的字符集合
  • 第一个参数指定一个字符串,它包含了0个或者多个由sep字符串中一个或者多个分隔符分割的标记。
  • strtok函数找到str中的下⼀个标记,并将其用 \0 结尾,返回一个指向这个标记的指针。(注:strtok函数会改变被操作的字符串,所以在使用strtok函数切分的字符串⼀般都是临时拷贝的内容并且可修改。)
  • strtok函数的第一个参数不为 NULL ,函数将找到str中第一个标记,strtok函数将保存它在字符串中的位置。
  • strtok函数的第一个参数为 NULL ,函数将在同一个字符串中被保存的位置开始,查找下一个标记。
  • 如果字符串中不存在更多的标记,则返回 NULL 指针。
#include <stdio.h>
#include <string.h>
int main()
{
	char arr[] = "192.168.6.111";
	char* sep = ".";
	char* str = NULL;
	for (str = strtok(arr, sep); str != NULL; str = strtok(NULL, sep))
	{
		printf("%s\n", str);
	}
	return 0;
}

12.strerror函数的使用

库中strtok的参数和返回类型:
char * strerror ( int errnum )
strerror函数可以把参数部分错误码对应的错误信息的字符串地址返回来。
在不同的系统和C语言标准库的实现中都规定了⼀些错误码,一般是放在 errno.h 这个头文件中说明的,C语言程序启动的时候就会使用一个全⾯的变量errno来记录程序的当前错误码,只不过程序启动的时候errno是0,表示没有错误,当我们在使用标准库中的函数的时候发生了某种错误,就会讲对应的错误码,存放在errno中,而一个错误码的数字是整数很难理解是什么意思,所以每一个错误码都是有对应的错误信息的。strerror函数就可以将错误对应的错误信息字符串的地址返回。

#include <errno.h>
#include <string.h>
#include <stdio.h>
//我们打印⼀下0~10这些错误码对应的信息
int main()
{
	int i = 0;
	for (i = 0; i <= 10; i++) 
	{
		printf("%s\n", strerror(i));
	}
	return 0;
}

在Windows11+VS2022环境下输出的结果如下:

No error
Operation not permitted
No such file or directory
No such process
Interrupted function call
Input/output error
No such device or address
Arg list too long
Exec format error
Bad file descriptor
No child processes

举例:

#include <stdio.h>
#include <string.h>
#include <errno.h>
int main ()
{
	FILE * pFile;
	pFile = fopen ("unexist.ent","r");
	if (pFile == NULL)
		printf ("Error opening file unexist.ent: %s\n", strerror(errno));
	return 0;
}

输出:

Error opening file unexist.ent: No such file or directory

也可以了解一下perror函数,perror函数相当于一次将上述代码中的第9行完成了,直接将错误信息打印出来。perror函数打印完参数部分的字符串后,再打印一个冒号和一个空格,再打印错误信息。

#include <stdio.h>
#include <string.h>
#include <errno.h>
int main ()
{
	FILE * pFile;
	pFile = fopen ("unexist.ent","r");
	if (pFile == NULL)
	perror("Error opening file unexist.ent");
	return 0;
}

输出:

Error opening file unexist.ent: No such file or directory

看完如果对你有帮助的话,别忘了一键三连哟!创作不易,谢谢大家呀!💞

版权声明:本文为博主作者:Fastrack527原创文章,版权归属原作者,如果侵权,请联系我们删除!

原文链接:https://blog.csdn.net/weixin_66058866/article/details/136790369

共计人评分,平均

到目前为止还没有投票!成为第一位评论此文章。

(0)
乘风的头像乘风管理团队
上一篇 2024年4月1日
下一篇 2024年4月1日

相关推荐