为什么当dest小于src但足够大以容纳src的子字符串时,strncpy()会产生垃圾?

9rygscc1  于 2023-10-16  发布在  其他
关注(0)|答案(3)|浏览(152)

我试图通过使用strncpy()来限制复制到dest(这里是str1)的n个字节的数量。dest对于n个字节来说已经足够大了,但是当dest小于source时,输出会产生垃圾(这里是argv[1])。这看起来不同,当我使dest足够大,以容纳源。
代码如下:

#include <stdio.h>                                                              
#include <string.h>                                                             
                                                                                
int main(int argc, char *argv[])                                                
{                                                                               
/* // Works when str1[?] is 20:                                                                    
  char str1[20];                                                                
  char str2[20]; */                                                             
                                                                                
  // produces garbage, str1 is smaller than argv[1] but big enough for the 4 bytes copied                                                            
  char str1[10];                                                                
  char str2[10];                                                                
                                                                                
  printf("argc = %i\n", argc);                                                  
  if (argc <= 1 || argc > 2){                                                   
    printf("Input Example:\n"                                                   
           "  %s SEGMENT:OFFSET\n"                                              
           "  %s 0100:F00A\n", argv[0], argv[0]);                               
    return 0;                                                                   
  }                                                                             
  printf("strlen(argv[1]) = %li, argv[1] = %s\n"                                
          , strlen(argv[1]), argv[1]);                                          
                                                                                
  // str1                                                                       
  strncpy(str1, argv[1], 4); // copying 4 bytes to str1                                                   
  printf("Copy only 4 Bytes -> sizeof(str1) = %li, "                            
         "strlen(str1) = %li, str1 = %s\n", sizeof(str1), strlen(str1), str1);  
                                                                                
  // str2                                                                       
  strncpy(str2, argv[1], 3); // copying 3 bytes to str2                                                   
  printf("Copy only 3 Bytes -> sizeof(str2) = %li, "                            
         "strlen(str2) = %li, str2 = %s\n", sizeof(str2), strlen(str2), str2);  
                                                                                
  return 0;                                                                     
}

0100:F00A的输入产生:

./test.bin 0100:F00A
argc = 2
strlen(argv[1]) = 9, argv[1] = 0100:F00A
Copy only 4 Bytes -> sizeof(str1) = 10, strlen(str1) = 8, str1 = 0100�U
Copy only 3 Bytes -> sizeof(str2) = 10, strlen(str2) = 3, str2 = 010

期望值为str1 = 0100
我也想知道为什么str2是正确的,它的初始数组大小和str1一样小。
当我改变

char str1[10]

char str1[20]

这样做,使它大于argv[1]的输入,那么输出是正确的:

./test.bin 0100:F00A
argc = 2
strlen(argv[1]) = 9, argv[1] = 0100:F00A
Copy only 4 Bytes -> sizeof(str1) = 20, strlen(str1) = 4, str1 = 0100
Copy only 3 Bytes -> sizeof(str2) = 20, strlen(str2) = 3, str2 = 010

看起来strncpy首先将所有内容复制到dest,然后删除其余内容。但这就是strncpy的工作方式吗?我假设它只是复制所需要的,4个字节。

z9gpfhce

z9gpfhce1#

来自手册页https://linux.die.net/man/3/strncpy
strncpy()函数与此类似,只是最多复制n个src字节。警告:如果src的前n个字节中没有空字节,则放置在dest中的字符串将不会以空结尾。
即在您的情况下没有空放在目的地

6l7fqoea

6l7fqoea2#

strncpy()的GLIBC手册页面:

警告:如果src的前n字节中没有空字节,则放入dest中的字符串将不会以空结尾。

你在观察效果。

wgx48brx

wgx48brx3#

不管它的价值,我试着编译和运行你的程序,因为是在我的PC(Windows/GCC 10.3.0).表面上看,它“看起来不错”。我没有看到你描述的艺术品。
这个例子说明了正在发生的事情:

#include <stdio.h>                                                              
#include <string.h>                                                             
                                                                                
int main(int argc, char *argv[])                                                
{ 
  char *s;
  char test1[] = "AAAAAAAA";                                                                             
  char test2[] = "AAAAAAAA";                                                                             

  printf("BEFORE: strlen(test1)=%lli, sizeof(test1)=%lli, test1=%s...\n",
    strlen(test1), sizeof(test1), test1);

  strncpy(test1, "BBBB", 4);
  printf("AFTER: strlen(test1)=%lli, sizeof(test1)=%lli, test1=%s...\n",
    strlen(test1), sizeof(test1), test1);

  strncpy(test2, "BBBB", 6);
  printf("AFTER: strlen(test2)=%lli, sizeof(test2)=%lli, test2=%s...\n",
    strlen(test2), sizeof(test2), test2);

  return 0;
}

  - gcc -o x -g -Wall -pedantic x.c
x.c:37:3: warning: 'strncpy' output truncated before terminating nul copying 4 bytes from a string of the same length [-Wstringop-truncation]
   37 |   strncpy(test1, "BBBB", 4);
      |   ^~~~~~~~~~~~~~~~~~~~~~~~~

  - x 0100:F00A
segment=0100, strlen(segment)=4; offset=F00A, strlen(offset)=4
BEFORE: strlen(test1)=8, sizeof(test1)=9, test1=AAAAAAAA...
AFTER: strlen(test1)=8, sizeof(test1)=9, test1=BBBBAAAA...
AFTER: strlen(test2)=4, sizeof(test2)=9, test2=BBBB...

附加注解:

  • 某些系统(BSD、Solaris等)提供strlcpy()

它最多将size-1字节复制到dest,始终添加一个终止空字节,并且不会用(更多)空字节填充目标。

  • 另一种将命令行解析为“segment”和offset”的方法是使用strtok()
...
if (argc != 2) {                                                   
  printf("USAGE: enter SEGMENT:OFFSET, e.g. \"0100:F00A\"\n");                               
  return 1;                                                                   
} 

if ((strlen(argv[1]) != 9) || (strchr(argv[1], ':') == NULL)) {
  printf ("This doesn't look like a segment::offset!\n");
  return 1;
}

s = strtok(argv[1], ":");
if (s == NULL){                                                   
  printf("Illegal entry: unable to parse %s\n", argv[1]);                               
} 
else {
  strcpy(segment, s);
  strcpy (offset, strtok(NULL, ":"));
  printf("segment=%s, strlen(segment)=%lli; offset=%s, strlen(offset)=%lli\n",
    segment, strlen(segment), offset, strlen(offset));
}

相关问题