2.8 strstr
char * strstr ( const char *str1, const char * str2);
strstr函数,是在str1主串中,查找是否存在子串str2,如果存在返回在str1主串中的位置(指针)
代码实现:
模拟实现strstr函数:
//模拟实现strstr char *my_strstr(const char* str1, const char* str2) { assert(str1 && str2);//断言 char* s1 = str1; char* s2 = str2; char* cp = str1;//创建三个指针 //第一个指针s1表示的是str1, //第二个指针s2表示的是str2,只是不断移动s1和s2 //第三个指针cp 是负责记录每一次循环的初始位置的 while (*s1) { s1 = cp; //变化cp while (*s1 == *s2&&*s1!='\0'&&*s2!='\0') { s1++; s2++; } if (*s2 == '\0') { return cp; } s2 = str2; cp++; } return cp; } int main() { char arr1[] = "abcabcdef"; char arr2[] = "abcd"; char *p=my_strstr(arr1, arr2); printf("%s\n", p); return 0; }
2.9 strtok
char * strtok ( char * str, const char * sep );
函数特性:
1.strtok函数找到str中的下一个标记,并将其用 \0 结尾,返回一个指向这个标记的指针。(注:strtok函数会改变被操作的字符串,所以在使用strtok函数切分的字符串一般都是临时 拷贝的内容并且可修改。)
2.strtok函数的第一个参数不为 NULL ,函数将找到str中第一个标记,strtok函数将保存它在 字符串中的位置。
3,strtok函数的第一个参数为 NULL ,函数将在同一个字符串中被保存的位置开始,查找下一 个标记。
4.如果字符串中不存在更多的标记,则返回 NULL 指针
代码演示:
2.10 strerror
char * strerror ( int errnum );
strerror函数:返回错误码,所对应的错误信息,错误码存储在errno上,且需要头文件<errno.h>
代码演示:
perror()函数和printf......strerror(errno) 是一样的作用,所以可以使用perror()来输出错误信息,fopen()是打开文件的函数,fclose()函数是关闭文件的函数
2.11 memcpy
内存操作函数
void * memcpy ( void * destination, const void * source, size_t num );
函数特征:
1.函数memcpy从source的位置开始向后复制num个字节的数据到destination的内存位置。
2.这个函数在遇到 '\0' 的时候并不会停下来。
3.如果source和destination有任何的重叠,复制的结果都是未定义的。
代码演示:
模拟实现:
//内存管理函数 //void * memcpy ( void * destination, const void * source, size_t num ); // //函数memcpy从source的位置开始向后复制num个字节的数据到destination的内存位置。 //这个函数在遇到 '\0' 的时候并不会停下来。 //如果source和destination有任何的重叠,复制的结果都是未定义的 #include<assert.h> //模拟实现memcpy内存管理函数 void* my_memcpy(void* dest, const void* src, int num) { assert(dest && src);//断言 void* ret = dest;//返回这个地址 while (num--) { //将src上面num个字节复制给dest对应的 位置 //字节的话 需要强制类型转换成 char*类型 *(char*)dest = *(char*)src; dest = (char*)dest + 1; src = (char*)src + 1; } return ret; } int main() { char arr1[] = "abcdef"; char arr2[] = "ccc"; memcpy(arr1, arr2,4); printf("%s\n", arr1); return 0; }
2.12 memmove
void * memmove ( void * destination, const void * source, size_t num );
函数特性:
1.和memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠的。
2.如果源空间和目标空间出现重叠,就得使用memmove函数处理
代码演示:
模拟实现memmove函数:
void*my_memmove(void* dest, const void* src, int num) { //使用memmove 主要是 处理重复的时候,如何复制的情况 //主要是处理好边界问题 //内存存储的时候,是低地址到高地址 //当dest > src的时候,可以是后面先传递,之后传递前面的 后->前 //当src < dest 的时候,发现前到后,后到前都可以,为了方便,我们统一使用前->后 assert(dest && src); //断言 void* ret = dest; if (dest < src) { while (num--) { //将src上面num个字节复制给dest对应的 位置 //字节的话 需要强制类型转换成 char*类型 *(char*)dest = *(char*)src; dest = (char*)dest + 1; src = (char*)src + 1; } } else { while (num--) {//如果num=5 *((char*)dest + num) = *((char*)src + num); //(char*)dest + num 表示在dest起始位置加上num个字节,得到dest之后的四个字节,为dest整体要移动的第五个位置,从0 1 2 3 4 正好五个 } } } return ret; } //memmove 函数 int main() { char arr1[] = "abcdef"; char arr2[] = "ccc"; my_memmove(arr1, arr1+2, 3); printf("%s\n", arr1); return 0; }
2.13 memcmp
int memcmp ( const void * ptr1,const void * ptr2,size_t num );
函数特性:
比较从ptr1和ptr2指针开始的num个字节
1.与strncmp函数类似,只是前者是只能用于字符串,后者memcmp可以用于任何类型的实参
2.num直接表示的是字节数,当ptr是整数数组的时候,就需要sizeof(ptr)当作num啦
代码实现:
三、字符函数
字符函数,包括字符分类函数,字符转换函数,头文件为<ctype.h>
3.1字符分类函数
字符分类函数:如果他的参数符合字符函数的要求就返回真,否则返回0
以下是字符分类函数的函数名和定义
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 | 任何可打印字符,包括图形字符和空白字符 |
代码演示(选择其中一个):
3.2 字符转换函数
int tolower ( int c ); 变小写
int toupper ( int c ); 变大写
代码演示:
总结
我们到这个地方就对于字符串函数、字符函数的基本上的函数有了大概的认识和了解,代码演示,以及模拟实现的一些字符串函数,这样希望对于小伙伴们,有所帮助,那么就点点关注,一键三连不迷路,这边预告,下一章节,咱来讲讲,关于结构体的相关知识。