字符函数和字符串函数

1.字符分类函数

用于判断字符是什么类型的函数,使用这些函.数都需要包含一个头文件ctype.h

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

    printf("%d\n",isdigit('1'));//十进制'0'-'9'字符
	printf("%d\n", isdigit('F'));//假--0

	printf("%d\n", islower('a'));//小写字母a-z
	printf("%d\n", islower('A'));//假--0
	
	printf("%d\n", isupper('A'));//大写字母A-Z
	printf("%d\n", isupper('a'));//假--0
	
	return 0;
}

写一个代码,将字符串中的小写字母转大写,其他字符不变

int main()
{
	char str[] = "I Am a Student";
	int i = 0;
	while (str[i] != '\0')
	{
		if (islower(str[i]))
		{
			str[i] = toupper(str[i]);
		}
		i++;
	}
	printf("%s\n", str);
	return 0;
}

2.字符转换函数

int tolower ( int c ); //将参数传进去的大写字母转小写

int toupper ( int c ); //将参数传进去的小写字母转大写

#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的使用和模拟实现

size_t strlen ( const char*  str );

判断下面的代码结果如何?

#include<stdio.h>
#include<string.h>
void func(char str1[], char str2[])
{
	if ((strlen(str1) - strlen(str2)) > 0)
        //strlen的返回类型是size_t,所以就算算出来是负数,
		//编译器也会默认该数是无符号整型,将其符号位作为数值位,得到一个很大的数
		//可以通过强制类型转化为int类型得到负数
		printf(">\n");
	else
		printf("<=\n");
}
int main()
{
	char str1[] = "abc";
	char str2[] = "abcdef";
	func(str1, str2);
	
	return 0;
}

strlen的模拟实现

//1.计数器
int my_strlen(char str[])
{
	int i = 0;
	int count = 0;
	while(str[i]!='\0')
	{
		count++;
		i++;
	}
	return count;
}
int main()
{
	char str[] = "abcdef";
	int len = my_strlen(str);
	printf("%d\n",len);
	return 0;
}

//2.指针-指针
size_t my_strlen(char* s)
{
	char* p = s;
	while (*s != '\0')
		s++;
	return s - p;
}
int main()
{
	char str1[] = "abced";
	size_t len = my_strlen(str1);
	printf("%zd\n", len);
	return 0;
}

//3.递归的方式(不创建临时变量,求字符串长度,一般不会直接说明用递归的方式)
size_t my_strlen(const char* s)
{
	int i = 0;
	if (*s == '\0')
		return 0;
	else
		return 1 + my_strlen(s + 1);
}
int main()
{
	char str[] = "abwdefg";
	size_t len = my_strlen(str);
	printf("%zd\n", len);
	return 0;
}

4.strcpy的使用和模拟实现

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

strcpy的模拟实现

//把source的内容拷贝到destination中
#include<assert.h>
//方法一
void my_strcpy(char* dest, const char* src)
{
	assert(dest != NULL);
	assert(src != NULL);
	while (*src != '\0')
	{
		*dest = *src;
		dest++;
		src++;
	}
	*dest = *src;
	
}
 
//方法二
void my_strcpy(char* dest, const char* src)
{
	assert(dest != NULL);
	assert(src != NULL);
	while (*dest++ = *src++)
		;
}
 
//方法三
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[] = "abcd";
	char str2[20];
	char* ret=my_strcpy(str2, str1);
	printf("%s\n", str2);
	printf("%s\n", ret);
	return 0;
}

5.strcat的使用和模拟实现

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

strcat的模拟实现

char* my_strcat(char* dest, char* src)
{
	assert(dest && src);
	char* ret = dest;
	while (*dest != '\0')
		dest++;
	while (*dest++ = *src++)
		;
	return ret;
}
int main()
{
	char str[20] = "abcde";
	char str1[] = "abc";
	char* ret=my_strcat(str, str1);
	printf("%s\n", str);
	printf("%s\n", ret);

	return 0;
}

6.strcmp的使用和模拟实现

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

int my_strcmp(const char* s1, const char* s2)
{
	assert(*s1 && s2);
	while (*s1 == *s2)
	{
		if (*s1 == '\0')
			return 0;
		s1++;
		s2++;
	}
	return *s1 - *s2;
}
int main()
{
	char str1[] = "abcd";
	char str2[] = "abcf";
	int ret = my_strcmp(str1, str2);
	printf("%d\n", ret);
	return 0;
}

7.strncpy函数的使用

char * strncpy ( char * destination, const char * source, size_t num );

拷⻉num个字符从源字符串到⽬标空间,如果源字符串的⻓度⼩于num,则拷⻉完源字符串之后,在⽬标的后边追加0,直到num个。

int main()
{
	char str[] = "sbcef";
	char str1[20] = "abxxxxxxxx";
	char* ret1 = strncpy(str1, str, 3);//只把sbc复制过来,覆盖原来的内容,没有带'\0'
	printf("%s\n", ret1);
	return 0;
}

8.strncat函数的使用

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

将source指向字符串的前num个字符追加到destination指向的字符串末尾,再追加⼀个 \0 字 符。如果source 指向的字符串的⻓度⼩于num的时候,只会将字符串中到 \0 的内容追加到destination指向的字符串末尾。

int main()
{
	char str[] = "sbcef";
	char str1[20] = "ab\0xxxx";
	char* ret = strncat(str1, str, 2);	//只复制sb,还会复制'\0'
	printf("%s\n", ret);
	return 0;
}

9.strcmp函数的使用

int strncmp ( const char * str1, const char * str2, size_t num );

⽐较str1和str2的前num个字符,如果相等就继续往后⽐较,最多⽐较num个字⺟,如果提前发现不⼀ 样,就提前结束,⼤的字符所在的字符串⼤于另外⼀个。如果num个字符都相等,就是相等返回0.

int main()
{
	char str[] = "sbcef";
	char str1[20] = "ab";
	int ret = strncmp(str1, str, 3);
	printf("%d\n", ret);
	return 0;
}

10.strstr的使用和模拟实现

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

函数返回字符串str2在字符串str1中第⼀次出现的位置。字符 串的⽐较匹配不包含 \0 字符,以 \0 作为结束标志。

int main()
{
	char str1[] = "this is an apple";
	char str2[] = "is";
	char* ret = strstr(str1, str2);
	printf("%s\n", ret);
	return 0;
}

strstr的模拟实现

char* mystrstr(const char* str1, const char* str2)
{
	char* s1 = NULL;
	char* s2 = NULL;
	char* cur = str1;
	while (*cur)
	{
		s1 = cur;
		s2 = str2;
		while (*s1!='\0'&&*s2!='\0'&& * s1 == *s2)
		{
			s1++;
			s2++;
		}
		if (*s2 == '\0')
			return (char*)cur;
		cur++;
	}
	return NULL;
}
int main()
{
	char str1[] = "this is an apple";
	char str2[] = "is";
	char* ret = mystrstr(str1, str2);
	if (ret == NULL)
		printf("没有找到\n");
	else
		printf("%s\n", ret);
	return 0;
}

11.strtok函数的使用

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

sep参数指向⼀个字符串,定义了⽤作分隔符的字符集合 ; 第⼀个参数指定⼀个字符串,它包含了0个或者多个由sep字符串中⼀个或者多个分隔符分割的标 记;strtok函数找到str中的下⼀个标记,并将其⽤ \0 结尾,返回⼀个指向这个标记的指针。(注: strtok函数会改变被操作的字符串,所以在使⽤strtok函数切分的字符串⼀般都是临时拷⻉的内容 并且可修改。); strtok函数的第⼀个参数不为 NULL ,函数将找到str中第⼀个标记,strtok函数将保存它在字符串 中的位置; strtok函数的第⼀个参数为 NULL ,函数将在同⼀个字符串中被保存的位置开始,查找下⼀个标 记; 如果字符串中不存在更多的标记,则返回 NULL 指针。

int main()
{
	char str[] = "2145348372234@qq.com";
	char str1[] = "@.";
	char* ret;

	for (ret = strtok(str, str1); ret != NULL; ret = strtok(NULL, str1))
	{
		printf("%s\n", ret);
	}

	/*ret = strtok(str, str1);
	printf("%s\n", ret);

	ret = strtok(NULL, str1);
	printf("%s\n", ret);

	ret = strtok(NULL, str1);
	printf("%s\n", ret);*/

	return 0;
}

12.strerror函数的使用

char * strerror ( int errnum );

strerror 函数可以把参数部分错误码对应的错误信息的字符串地址返回来。在不同的系统和C语⾔标准库的实现中都规定了⼀些错误码,⼀般是放在 errno.h 这个头⽂件中说明的。strerror函数就可以将错误对应的错误信息字符串的地址返回。

#include <errno.h>
#include <string.h>
#include <stdio.h>
int main()
{
	int i = 0;
	for (; 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

#include <stdio.h>
#include <string.h>
#include <errno.h>
int main()
{
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		printf("Error opening file unexist.ent: %s", strerror(errno));
	}
	return 0;
}
//结果为:Error opening file unexist.ent: No such file or directory

int main()
{
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror("Error opening file unexist.ent");
        //perror函数打印完参数部分的字符串后,再打印⼀个冒号和⼀个空格,再打印错误信息。
	}
	return 0;
}
//结果为:Error opening file unexist.ent: No such file or directory
  • 40
    点赞
  • 49
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值