鱼C论坛

 找回密码
 立即注册
查看: 3270|回复: 2

[已解决]S1E22动动手第1题:用指针1拷贝字符串 —— strncpy 函数

[复制链接]
发表于 2021-4-22 13:38:05 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
本帖最后由 weijun_zhang 于 2021-4-22 13:41 编辑

S1E22动动手第1题进阶题:用指针1拷贝字符串 —— strncpy 函数
请问下方函数为何错误呢?  

  1. #include <stdio.h>

  2. int main()
  3. {
  4.        char str1[1028],str2[1028];
  5.        char *target1=str1;
  6.        char *target2=str2;
  7.        int n;
  8.       
  9.        printf("请输入一个字符串到str1中:");
  10.            fgets(str1,1028,stdin);
  11.        printf("请输入需要拷贝的字符个数:");
  12.                scanf("%d",&n);
  13.        printf("开始拷贝。。。\n");        
  14.           // printf("str2的内容是:");   
  15.            while(n>0)
  16.            {

  17.                            if(*target1=='\0')  //此处是==,不是=
  18.                       {
  19.                                   break;
  20.                            }
  21.             if((int)(*target1)<0)    //汉字占2个字符的处理
  22.                {
  23.                          *target2=*target1;
  24.                                target1++;
  25.                    target2++;
  26.                   *target2=*target1;
  27.                                    target1++;
  28.                                    target2++;
  29.                            }
  30.             else
  31.                    {
  32.                           *target2=*target1;
  33.                           target1++;
  34.               target2++;
  35.                         }
  36.                           n--;
  37.                    }

  38.         printf("str2的内容是:%s",str2);

  39.         return 0;
  40. }
复制代码


下方为老师的代码
  1. #include <stdio.h>

  2. #define MAX 1024

  3. int main()
  4. {
  5.         char str1[MAX];
  6.         char str2[MAX];

  7.         char *target1 = str1;
  8.         char *target2 = str2;

  9.         char ch;
  10.         int n;

  11.         printf("请输入一个字符串到 str1 中:");
  12.         fgets(str1, MAX, stdin);

  13.         printf("请输入需要拷贝的字符个数:");
  14.         scanf("%d", &n);

  15.         printf("开始拷贝 str1 的内容到 str2 中...\n");
  16.         while (n--)
  17.         {
  18.                ch = *target2++ = *target1++;
  19.                if (ch == '\0')
  20.                {
  21.                        break;
  22.                }
  23.                if ((int)ch < 0)
  24.                {
  25.                        *target2++ = *target1++;
  26.                        *target2++ = *target1++;
  27.                }
  28.         }

  29.         *target2 = '\0';

  30.         printf("拷贝完毕!\n");
  31.         printf("现在,str2 中的内容是:%s\n", str2);

  32.         return 0;
  33. }
复制代码
最佳答案
2023-4-30 10:20:25
这段代码中的错误在于,当源字符串 str1 中需要拷贝的字符个数 n 大于实际字符个数时,目标字符串 str2 仍然没有以 \0 结尾。这将导致在输出目标字符串 str2 的内容时,会出现意料之外的结果,并且可能会访问到未分配的内存空间而产生不可预测的错误。

正确的方法是,在拷贝完成后,手动向目标字符串 str2 的结尾位置添加 \0,即用 target2 指针指向目标字符串 str2 的最后一个字符,然后将其赋值为 '\0'。这样才能保证目标字符串 str2 的有效性,避免不必要的错误。

以下是修改后的示例代码:

  1. #include <stdio.h>
  2. #include <string.h>

  3. int main()
  4. {
  5.     char str1[1028], str2[1028];
  6.     char *target1 = str1;
  7.     char *target2 = str2;
  8.     int n;

  9.     printf("请输入一个字符串到 str1 中:");
  10.     fgets(str1, 1028, stdin);
  11.     printf("请输入需要拷贝的字符个数:");
  12.     scanf("%d", &n);
  13.     printf("开始拷贝。。。\n");

  14.     while (n > 0 && *target1 != '\0') {
  15.         if ((int)(*target1) < 0) {
  16.             // 汉字占两个字符
  17.             *target2++ = *target1++;
  18.             *target2++ = *target1++;
  19.         } else {
  20.             *target2++ = *target1++;
  21.         }
  22.         n--;
  23.     }

  24.     // 手动添加结尾标志
  25.     *target2 = '\0';

  26.     printf("str2 的内容是:%s\n", str2);

  27.     return 0;
  28. }
复制代码

修改后的代码使用了 C 标准库中的 string.h 头文件中提供的字符串函数,避免了手动处理 \0 结尾标志的繁琐过程。同时,为了确保源字符串和目标字符串的安全性,还需要在拷贝时判断源字符串是否已经结束(即 *target1 == '\0'),如果结束了就退出循环,否则继续拷贝。
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

 楼主| 发表于 2021-4-22 13:41:07 | 显示全部楼层
我已经自己搞定啦。  
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2023-4-30 10:20:25 | 显示全部楼层    本楼为最佳答案   
这段代码中的错误在于,当源字符串 str1 中需要拷贝的字符个数 n 大于实际字符个数时,目标字符串 str2 仍然没有以 \0 结尾。这将导致在输出目标字符串 str2 的内容时,会出现意料之外的结果,并且可能会访问到未分配的内存空间而产生不可预测的错误。

正确的方法是,在拷贝完成后,手动向目标字符串 str2 的结尾位置添加 \0,即用 target2 指针指向目标字符串 str2 的最后一个字符,然后将其赋值为 '\0'。这样才能保证目标字符串 str2 的有效性,避免不必要的错误。

以下是修改后的示例代码:

  1. #include <stdio.h>
  2. #include <string.h>

  3. int main()
  4. {
  5.     char str1[1028], str2[1028];
  6.     char *target1 = str1;
  7.     char *target2 = str2;
  8.     int n;

  9.     printf("请输入一个字符串到 str1 中:");
  10.     fgets(str1, 1028, stdin);
  11.     printf("请输入需要拷贝的字符个数:");
  12.     scanf("%d", &n);
  13.     printf("开始拷贝。。。\n");

  14.     while (n > 0 && *target1 != '\0') {
  15.         if ((int)(*target1) < 0) {
  16.             // 汉字占两个字符
  17.             *target2++ = *target1++;
  18.             *target2++ = *target1++;
  19.         } else {
  20.             *target2++ = *target1++;
  21.         }
  22.         n--;
  23.     }

  24.     // 手动添加结尾标志
  25.     *target2 = '\0';

  26.     printf("str2 的内容是:%s\n", str2);

  27.     return 0;
  28. }
复制代码

修改后的代码使用了 C 标准库中的 string.h 头文件中提供的字符串函数,避免了手动处理 \0 结尾标志的繁琐过程。同时,为了确保源字符串和目标字符串的安全性,还需要在拷贝时判断源字符串是否已经结束(即 *target1 == '\0'),如果结束了就退出循环,否则继续拷贝。
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-6-10 11:43

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表