字符串函数的重点:
文章不长,是为了让你一点点消化所有内容:
strncpy函数
先来看一下,strncpy函数的声明:
char * strncpy ( char * destination, const char * source, size_t num );
与strcpy函数相比,strncpy函数只是多了一个参数:size_t num,也就是需要复制的长度
来看一下下面的例子:
这里是将arr2中的字符串拷贝到arr1中,指定拷贝4个,arr2不是刚好有4个字符吗,为什么会出现这样的结果呢?
注意看右边,arr2末尾还有一个\0未拷贝过去
如图:
当我们将 拷贝长度4改成5时,就可以完成了
所以说,strncpy函数还是比较乖的,我们让他拷贝几个,他就拷贝几个,但是,看到这里,它是真的乖吗?
再看下面,假如我把长度5改成长度10呢
可以发现,strncpy函数不仅帮我们将第五个改成了\0,还将超出arr2本身的长度那一部分,都改成了\0,所以,这是乖还是懂事还是自作聪明,留给你进一步探讨。
再有一个问题:
我们刚开始是将arr2拷贝到arr1中,很明显,arr2的长度小于arr1的长度,但是当我们将arr1的长度拷贝到arr2中呢?
int main() { char arr1[] = "hello world"; char arr2[] = "qwer"; strncpy(arr2, arr1, sizeof(arr1)); //这里只是为了演示目标空间不够大,才写sizeof(arr1) printf("%s\n", arr1); }
实践出真知:
翻译可得,意思就是,所拷贝的内容超出了arr2的数组的范围,造成数组越界了。
所以,strcpy的注意事项
有几点:
模拟实现strncpy
char* my_strncpy(char* arr1, const char* arr2, int sz) { assert(arr1 && arr2); char* ret = arr1; while (sz &&(*arr1++ = *arr2++)!= '\0') { sz--; } if (sz) { while (--sz) //注意,如果是sz--,会多更改一次'\0' { *arr1++ = '\0'; //根据strncpy函数的分析,多余的长度全部要改写成\0 } } return ret; } int main() { char arr1[] = "hello world"; char arr2[] = "qwer"; char*ret = my_strncpy(arr1, arr2, 10); printf("%s\n", ret); }
情况1:当arr1 = ‘\0’时,意味着arr2已经全部拷贝到arr1中
情况2:当sz=0时,已经完成拷贝
上面的代码是情况1,如果想出现情况2,只需将所需要拷贝的长度更改到小于源数组的长度
结果如上:
当我们设置成–sz时,会多更改一次’\0’,结果如下:,虽然打印出来不会改变,但是内部已经发生改变
arr1[10]已经被更改成了 ‘\0’
2.strncmp函数的详解
先来看一下strncmp函数的声明:
int strncmp(const char* str1, const char* str2, size_t num);
与strcmp函数相比,strncmp函数只是多了一个参数:size_t num,也就是所需要相比的字节的个数。
来到例题感受一下:
int main() { char arr1[] = "abcd"; char arr2[] = "abcdef"; int ret =strncmp(arr1, arr2, 3); printf("%d\n", ret); }
我们需要比较strncmp函数的前面三个字节,(由于一个字符大小是一个字节),即比较前三个字符的大小,很明显,arr1和arr2中的前三个字符的大小都相等
由上图,当 arr1第一个字符 - arr2第一个字符时,若<0,则返回一个<0的数字,若>0,则返回一个>0的数字,若相等,则返回0;
所以结果一目了然。
当我们比较前5个字节时,很明显,arr1
模拟实现strncmp函数
int my_strncmp(const char* str1, const char* str2, unsigned int num) { assert(str1 && str2); while (num-- && *str1 && *str1 == *str2) 三种退出循环的情况 { 1.num退出循环后,俩字符串相等 2.*str1=='\0'时,可能相等,可能不相等 3.*str1 !=*str2,必不相等 str1++; str2++; } return *str1 - *str2; 不管哪种情况,退出循环之后,*str1 - *str2都能满足要求 }
重要代码部分已加解释。
下面来看一下结果:
这里返回-101的原因是,比较第五个字符(即第五个字节)时,arr1中的第五个字符是’\0’,arr2中的第五个字符是e,'\0’对应的ascii码值是0,e对应的ascii码值是101, 0-101 = -101
不管值为多少,库中的strncmp返回-1,模拟的strncmp返回-101,都小于0,都能够比较两字符串的大小。但是在visual studio环境下,strncmp和strcmp函数对于小于0或者大于0的数字,统一返回-1和1
比较一下,证明了上述的结论。所以,在模拟实现该函数时,个人认为,具体到返回的值为两个字符之间的差,更易于理解。
因为由差可以得出具体到哪两个字符不相等,差值是多少。
注意事项 -
1.目标空间必须足够大,
2.目标空间必须可修改
3.源字符串必须以’\0’结束
3.strncat函数的巧解
char* strncat(char* destination, const char* source, size_t num);
与strcat函数相比,strncat函数仍然是多了一个参数:size_t,这个参数是说:追加的字节个数。
注意,函数的追加是在\0后面追加的
来看例题感受一下:
int main() { char arr1[20] = "hello"; char arr2[] = "world"; strncat(arr1, arr2, 3); }
在arr1的\0后面追加3个字节,这三个字节来源于arr2,追加结果就是 “hellowor” ,后面默认加上了\0。
那为什么会自己在\0后面追加呢?
看一下库函数的介绍,就是在 \0 后面追加字符
注意:(1)当num 小于 源字符串的长度时,库函数strncat会主动加上’\0’,假如这里 num = 3,追加的时候,从arr1中的后面的\0开始追加,追加三个,即hellowor
但是在r后面,strncat会主动加上一个,就一个\0,至于后面有没有追加到num个,strncat也不管了
(2)当num 大于 源字符串的长度时,库函数strncat ,假如num = 8 ,尽管num大于arr2的长度,strncat仍然在追加完成后,主动加一个\0,且仅加一个,后面的也不管了。
我们来验证一下:
(1)当num小于源字符串时,会主动在最后面加上一个\0,且只加一个。
(2)当num大于源字符串时,会主动在最后面加上一个\0,且只加一个。
总结:
不管num大于还是小于 源字符串的长度,strncat都会主动在最后面加上一个 \0, 注意:就加一个
了解了库中的strncat函数后,我们来模拟实现my_strncat 函数
3.1模拟实现my_strncat函数
char* my_strncat(char* dest, const char* src, unsigned int num) { assert(dest && src); char* ret = dest; while (*dest++) //不使用++*dest是因为,假如dest是一个空字符串,进入循环之后,就已经跳过了'\0',造成越界访问,发生意外 { ;//找到目的地字符串的\0 } dest--;//退出循环后,dest指向了'\0'的后一位,所以需要dest-- while (num--) { if ((*dest++ = *src++) == 0) { return ret;//意味着源字符串遇到\0了,已经追加完成。但是num未到0 } } *dest = '\0'; // 退出循环后,表明num的值不为正数了,此时dest指向了'\0'的后一个位置,将此位置置为'\0' return ret; }
可以看到,结果与预期相符。
注意:
- 1.目标空间必须足够大,
2.目标空间必须可修改
3.源字符串必须以’\0’结束
一次性看到这里,你需要回去消化一下上面的内容,不然你会吃不消,剩下的重点,我们下期见!
看到这里,如果你觉得对你有帮助,不妨关注一下,持续为你输出更高质量的知识。