本内容是重点介绍使用字符函数、字符串函数、内存函数的库函数的使用和注意事项。
前言1.strlen求字符串长度 字符串查找
strlen strstr
长度不收限制的字符串函数 错误信息报告
strcpy strerror
strcat 内存操作函数
strcmp memcpy
长度受限制的字符串函数介绍 memmove
strncpy memset
strncat memcmp
strncmp 字符操作
:求字符串长度
注意事项:size_t strlen (const char * str);
- 字符串已' '作为结束标志,strlen函数返回的是在字符串中' '前面出现的字符个数(不包含 )
- 参数指向的字符串必须要以' ' 结束
- 注意函数的返回值是size_t (unsigned int),是无符号的
#include模拟实现strlen函数#include int main() { char arr1[]= "abcdef"; int ret = strlen(arr1); printf("%dn", ret); return 0; }
#include2. strcpysize_t my_strlen(const char* str) { char* st = str; while (*st++); return st-str-1; } int main() { char arr1[] = "abcdef"; int ret = my_strlen(arr1); printf("%dn", ret); }
: 字符串拷贝
注意事项:char* strcpy(char * destination, const char * source)
- 源字符串必须以 ' ' 结束。
- 会将源字符串中的 ' ' 拷贝到目标空间。
- 目标空间必须足够大,以确保能存放源字符串。
- 目标空间必须可变。
#include模拟实现strcpy:#include int main() { char arr1[20] = { 0 }; char arr2[] = "abcdef"; printf("%sn",strcpy(arr1, arr2)); return 0; }
#include3.strcat#include char* my_strcpy(char* dest, const char* src) { assert(dest && src); char* ret = dest; while (*dest++ = *src++) { ; } return ret; } int main() { char arr1[20] = { 0 }; char arr2[10] = "abcdef"; printf("%sn", my_strcpy(arr1,arr2)); return 0; }
:字符串连接(字符串追加)
注意事项:char * strcat ( char * destination, const char * source)
- 源字符串必须以 ' ' 结束。
- 目标空间必须有足够的大,能容纳下源字符串的内容。
- 目标空间必须可修改。
#include模拟实现strcat:#include int main() { char a1[20] = "abc"; char a2[] = "def"; printf("%sn", strcat(a1, a2)); return 0; }
#include4.strcmpchar* my_strcat(char* dest, const char* src) { char* ret = dest; while (*dest) { dest++; } while (*dest++ = *src++) { ; } return ret; } int main() { char a1[20] = "abc"; char a2[] = "def"; printf("%sn", my_strcat(a1, a2)); }
:字符串比较,比较的是字符串的内容,不是长度
标准规定:int strcmp ( const char * str1,const char * str2);
- 第一个字符串大于第二个字符串,则返回大于0的数字
- 第一个字符串等于第二个字符串,则返回0
- 第一个字符串小于第二个字符串,则返回小于0的数字
#include模拟实现strcmp:#include int main() { char a1[] = "abcd"; char a2[] = "abcd"; int ret = strcmp(a1, a2); printf("%dn", ret); return 0; }
#include5.strncpyint my_strcmp(const char* s1, const char* s2) { while (*s1 == *s2) { if (*s1 == ' ') { return 0; } s1++; s2++; } return *s1 - *s2; } int main() { char a1[] = "abcd"; char a2[] = "abcd"; int ret = my_strcmp(a1, a2); printf("%dn", ret); return 0; }
注意事项:char * strncpy ( char * destination , const char * source , size_t num);
- 拷贝num个字符从源字符串到目标空间。
- 如果源字符串的长度小于num,则拷贝完源字符串之后,在目标的后边追加0,直到num个。
#include6.strcat#include int main() { char arr1[20] = { 0 }; char arr2[] = "abcdef"; printf("%sn", strncpy(arr1, arr2 + 2, 2));//输出结果cd return 0; }
实例:char * strncat ( char * destination , const char * source ,size_t num);
#include7.strncmp#include int main() { char str1[20]; char str2[20]; strcpy(str1, "To be "); strcpy(str2, "or not to be"); strncat(str1, str2, 6); puts(str1);//输出结果,To be or not return 0; }
注意事项:int strncmp ( const char * str1, const char * str2, size_t num );
比较到出现另个字符不一样或者一个字符串结束或者num个字符全部比较完。
实例:#include8. strstr#include int main() { char str[][5] = { "R2D2","C3PO ","R2A6" }; int n; puts("Looking for R2 astromech droids..."); for (n = 0; n < 3; n++) if (strncmp(str[n], "R2xx", 2) == 0) { printf("found %sn", str[n]);//输出结果 R2D2 R2A6 } return 0; }
:字符串查找函数 在一个字符串中查找子字符串
实例:char * strstr ( const char * str2, const char * str1);
#include模拟实现strstr:#include int main() { char arr1[] = "i am a good student,hehe student"; char arr2[] = "student"; //查找arr1中arr2第一次出现的位置 char* ret = strstr(arr1, arr2); if (ret == NULL) { printf("找不到n"); } else { printf("%sn", ret);//输出 student,hehe student } return 0; }
#include9.strtok#include char* my_strstr(const char* str1, const char* str2) { assert(str1 && str2); char* s1; char* s2; char* cp = str1; if (*str2 == ' ') { return str1; } while (*cp) { s1 = cp; s2 = str2; while (*s1 != ' ' && *s2 != ' ' && *s1 == *s2) { s1++; s2++; } if (*s2 == ' ') { return cp; } cp++; } return NULL; } int main() { char a1[] = "hello my name is wz zz"; char a2[] = "wz"; char* ret = my_strstr(a1, a2); if (ret == NULL) { printf("找不到了n"); } else { printf("%sn", ret); } return 0; }
:分割字符串
注意事项 :char * strtok ( char * str, const char * sep );
- sep参数是个字符串,定义了用作分隔符的字符集合
- 第一个参数指定一个字符串,它包含了0个或者多个由sep字符串中一个或者多个分隔符分割的标 记。
- strtok函数找到str中的下一个标记,并将其用 结尾,返回一个指向这个标记的指针。(注: strtok函数会改变被操作的字符串,所以在使用strtok函数切分的字符串一般都是临时拷贝的内容 并且可修改。)
- strtok函数的第一个参数不为 NULL ,函数将找到str中第一个标记,strtok函数将保存它在字符串 中的位置。
- strtok函数的第一个参数为 NULL ,函数将在同一个字符串中被保存的位置开始,查找下一个标 记。
- 如果字符串中不存在更多的标记,则返回 NULL 指针。
#include10.strerror#include int main() { char arr1[] = "wzzz@bwwwu.errrh"; char arr2[100] = { 0 };//临时数据 //zpw bitedu.tech char sep[] = "@."; strcpy(arr2, arr1); char* ret = NULL; //分割字符串 for (ret=strtok(arr2, sep); ret!=NULL; ret=strtok(NULL, sep)) { printf("%sn", ret); } //strtok(arr2, sep); //strtok(NULL, sep); return 0; }
char * strerror(int errnum);
返回误码,所对应的错误信息 。
实例:#include11.memcpy#include int main() { printf("%sn", strerror(0)); printf("%sn", strerror(1)); printf("%sn", strerror(2)); printf("%sn", strerror(3)); return 0; }
:拷贝的整形数据
注意事项:void * memcpy ( void * destination, const void * source, size_t num );
- 函数memcpy从source的位置开始向后复制num个字节的数据到destination的内存位置。
- 这个函数在遇到 ' ' 的时候并不会停下来。
- 如果source和destination有任何的重叠,复制的结果都是未定义的。
#include模拟实现memcpy:#include struct { char name[40]; int age; } person, person_copy; int main() { char myname[] = "Pierre de Fermat"; memcpy(person.name, myname, strlen(myname) + 1); person.age = 46; memcpy(&person_copy, &person, sizeof(person)); printf("person_copy: %s, %d n", person_copy.name, person_copy.age); return 0; }
#include12.memmove#include void* my_memcpy(void* dest, const void* src, size_t count) { void* ret = dest; assert(dest && src); while (count--) { *(char*)dest = *(char*)src; dest = (char*)dest+1; src = (char*)src+1; } return ret; } int main() { int arr[10] = { 1,2,3,4,5,6,7,8,9,10 }; int i = 0; my_memcpy(arr + 2, arr, 16); for (i = 0; i < 10; i++) { printf("%d ", arr[i]); } return 0; }
:内存拷贝时,出现内存重复的现象
注意事项:void * memmove ( void * destination, const void * source, size_t num );
- 和memcpy的差别就是memmove函数处理的源内存块和目标内存块是可以重叠的。
- 如果源空间和目标空间出现重叠,就得使用memmove函数处理。
#include模拟实现memmove:#include int main () { char str[] = "memmove can be very useful. ........... "; memmove (str+20,str+15,11); puts (str); return 0; }
#include13.memcmp#include void* my_memmove(void* dest, const void* src, size_t count) { void* ret = dest; assert(dest && src); if (dest < src) { //dest在src左边时,从前往后拷贝 while (count--) { *(char*)dest = *(char*)src; dest = (char*)dest + 1; src = (char*)src + 1; } return ret; } else { //dest在src右边时,从后往前拷贝 while (count--) { *((char*)dest+count) = *((char*)src+count); } return ret; } } int main() { int arr[10] = { 1,2,3,4,5,6,7,8,9,10 }; int i = 0; my_memmove(arr + 2, arr, 16); for (i = 0; i < 10; i++) { printf("%d ", arr[i]); } return 0; }
注意事项:int memcmp ( const void * ptr1, const void * ptr2, size_t num );
- 比较从ptr1和ptr2指针开始的num个字节
- 返回值:
#include14. memset#include int main() { int arr1[] = { 1,2,3,4,5 }; int arr2[] = { 1,2,3,6,6 }; int ret = memcmp(arr1, arr2, 12); printf("%dn", ret);//输出结果为 0 return 0; }
实例:void * memset(void *dest,int c,size_t count);
#include#include int main() { int arr[] = { 1,2,3,4,5,6 }; memset(arr, 1, 20); return 0; }



