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