字符串函数剖析(2)

简介: 字符串函数的重点:文章不长,是为了让你一点点消化所有内容:在这里插入图片描述

字符串函数的重点:

文章不长,是为了让你一点点消化所有内容:

strncpy函数

先来看一下,strncpy函数的声明:

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

与strcpy函数相比,strncpy函数只是多了一个参数:size_t num,也就是需要复制的长度

来看一下下面的例子:1fa3a37f65014e649edb173e84e54ec2.png

这里是将arr2中的字符串拷贝到arr1中,指定拷贝4个,arr2不是刚好有4个字符吗,为什么会出现这样的结果呢?


image.gif

注意看右边,arr2末尾还有一个\0未拷贝过去

9d3cf6c146ac49f699e099ee43947847.png

如图:

当我们将 拷贝长度4改成5时,就可以完成了


ff6bcbeb33704fd6aceb6ea2b5850b83.png

所以说,strncpy函数还是比较乖的,我们让他拷贝几个,他就拷贝几个,但是,看到这里,它是真的乖吗?

再看下面,假如我把长度5改成长度10呢

9ed2478219a947ecae4c8577faebb45a.png

可以发现,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);
}

实践出真知:


22519cc686794f629b733b41e981bec1.png

c947a353ebe64d01aa6877f2af124da0.png

翻译可得,意思就是,所拷贝的内容超出了arr2的数组的范围,造成数组越界了。

所以,strcpy的注意事项

有几点:

894e2f989ff94e7b9e598aa3c2424c8a.png

模拟实现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,只需将所需要拷贝的长度更改到小于源数组的长度

f9e995f8f6534f12bc884c77b03e5859.png

结果如上:

当我们设置成–sz时,会多更改一次’\0’,结果如下:,虽然打印出来不会改变,但是内部已经发生改变

fc9b4b2ac548487db636482743e22792.png

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中的前三个字符的大小都相等

2c40574ce8214dada030288698b1472c.png

由上图,当 arr1第一个字符 - arr2第一个字符时,若<0,则返回一个<0的数字,若>0,则返回一个>0的数字,若相等,则返回0;


f69fecf1b84a4fa0971f846e4d08c1f3.png

所以结果一目了然。

当我们比较前5个字节时,很明显,arr1

6899a780194141909fdd36c00dab560f.png

模拟实现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都能满足要求
}

重要代码部分已加解释。

下面来看一下结果:

4d8bdc10672047e1b6acd8c3543285b4.png

这里返回-101的原因是,比较第五个字符(即第五个字节)时,arr1中的第五个字符是’\0’,arr2中的第五个字符是e,'\0’对应的ascii码值是0,e对应的ascii码值是101, 0-101 = -101

image.png

不管值为多少,库中的strncmp返回-1,模拟的strncmp返回-101,都小于0,都能够比较两字符串的大小。但是在visual studio环境下,strncmp和strcmp函数对于小于0或者大于0的数字,统一返回-1和1

ecbf35ea8ce841bdadf8f96890800d1b.png

39e47d2bbf7748a7b6fe7c3d9b080a26.png

比较一下,证明了上述的结论。所以,在模拟实现该函数时,个人认为,具体到返回的值为两个字符之间的差,更易于理解。

因为由差可以得出具体到哪两个字符不相等,差值是多少。

注意事项 -

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 后面追加字符


1eb7abd0d9464c09946b1a32cf7c8e20.png

注意:(1)当num 小于 源字符串的长度时,库函数strncat会主动加上’\0’,假如这里 num = 3,追加的时候,从arr1中的后面的\0开始追加,追加三个,即hellowor

但是在r后面,strncat会主动加上一个,就一个\0,至于后面有没有追加到num个,strncat也不管了

(2)当num 大于 源字符串的长度时,库函数strncat ,假如num = 8 ,尽管num大于arr2的长度,strncat仍然在追加完成后,主动加一个\0,且仅加一个,后面的也不管了。

我们来验证一下:


a451e7b67041460aa51c5c0bea31ad2c.png

(1)当num小于源字符串时,会主动在最后面加上一个\0,且只加一个。

9dab0a6296d343f492a3320814b6f8fa.png

(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;
}

921b6027c8ed471abc549eaf82134753.png

可以看到,结果与预期相符。

注意:

- 1.目标空间必须足够大,
2.目标空间必须可修改
3.源字符串必须以’\0’结束

一次性看到这里,你需要回去消化一下上面的内容,不然你会吃不消,剩下的重点,我们下期见!

看到这里,如果你觉得对你有帮助,不妨关注一下,持续为你输出更高质量的知识。

相关文章
|
1月前
|
安全 C语言 C++
|
6月前
|
安全 C语言
需要知道的字符串函数
需要知道的字符串函数
|
存储 C语言
字符串函数介绍&应用(二)
字符串函数介绍&应用
|
存储 C语言
字符串函数介绍&应用(一)
字符串函数介绍&应用
|
6月前
字符串函数
字符串函数
|
编译器
C详细的字符串函数
C详细的字符串函数
67 0
|
11月前
详解字符函数和字符串函数-1
详解字符函数和字符串函数
41 0
|
编译器 C语言 Python
字符函数和字符串函数(上)
字符函数和字符串函数(上)
46 0
|
C语言
一文带你玩转C库中的一系列字符串函数
一文带你玩转C库中的一系列字符串函数
66 2
|
C语言
字符函数和字符串函数详解(一)
字符函数和字符串函数详解
47 0