即使我尝试在 C 中创建文件,也找不到文件时出现分段错误 固定代码

问题描述

我正在编写将文本附加到文件代码。 它在 fopen 的开头使用 WritetoFile,因此即使文件不存在,它也会创建它。

但是,当我根本没有输入文件时会发生什么?根本没有争论?只是./a.out

分段错误

我不知道为什么,在我看来,我所做的一切都很好,以避免出现任何问题。

int main(int argc,char **argv)
{
    
    char file_name[30] = "file.txt";
    
    printf("%s",file_name); /* for debugging : doesn't print it */
    
    if (0 != argc)
    {
        strcpy(file_name,argv[1]);
    }
    
    WritetoFile(file_name);
    
}

或 (以防我真的无法将字符串文字放入 char 数组中):

char file_name[30];
    
    if (0 == argc)
    {
       strcpy(file_name,"file.txt");
    }
    else
    {
        strcpy(file_name,argv[1]);
    }

对于我遇到的两种情况

Segmentation fault (core dumped)

解决方法

if (0 != argc)

argc 值通常是(a) 参数的计数包括程序名称。因此,运行 ./a.outargc 将为 1 而不是零。而且,由于 argv[argc] 通常是 NULL,取消引用它不会有好的结局:-)

如果你想确保另一个参数可用,你可以使用:

if (argc > 1) {            // Have both argv[0] AND argv[1].
    nowSafeToUse(argv[1]);
}

更详细地说,C11 标准规定:

如果声明了它们,则主函数的参数应遵守以下约束:

  • argc 的值应为非负。
  • argv[argc] 应为空指针。
  • 如果 argc 的值大于零,则数组成员 argv[0]argv[argc-1](包括在内)应包含指向字符串的指针,这些字符串由宿主环境事先赋予实现定义的值程序启动。目的是从托管环境中的其他地方向程序启动之前确定的程序提供信息。如果宿主环境无法提供包含大写和小写字母的字符串,则实现应确保以小写形式接收字符串。
  • 如果argc的值大于零,则argv[0]指向的字符串代表程序名;如果宿主环境中没有程序名,argv[0][0] 应为空字符。如果 argc 的值大于 1,则 argv[1]argv[argc-1] 指向的字符串表示程序参数。
  • 参数 argcargv 以及 argv 数组指向的字符串应可由程序修改,并在程序启动和程序终止之间保留它们最后存储的值。

关于这条线的旁白:

printf("%s",file_name); /* for debugging : doesn't print it */

如果这不是打印,可能是因为标准输出是行缓冲的(如果它被确定为交互式设备,则默认,否则完全缓冲)。

最后不输出 \n,字符可能仍位于某个缓冲区中,准备写入。崩溃可能会在不刷新缓冲区的情况下终止进程。所以一个简单的解决方案可能只是使用以下之一:

printf("%s",file_name);
puts(file_name);

作为另一个,如果您输入的文件名大于 29 个字符,您会遇到麻烦,因为它会溢出 file_name,允许 \0最后也一样。

更好的方法可能是直接使用您的默认字符串或 argv[1](无需复制),例如:

int main(int argc,char **argv) {
    char *file_name = (argv > 1)
        ? argv[1]
        : "file.txt";
    printf("%s\n",file_name); // for debugging : probably does print it :-)
    
    WriteToFile(file_name);
}

(a) 标准不要求,因为它允许特定于实现的差异,但这是通常的情况。具体来说,这句话:

... 在程序启动之前由宿主环境赋予实现定义的值。目的是从托管环境中的其他地方向程序启动之前确定的程序提供信息。

几乎意味着它可以为所欲为:-)

可以在 here 中找到相关答案(尽管是非重复问题)。

,

argc 为 1 时,argv 被视为包含 2 个指向 char 的指针的数组。第一个是有效的指针(通常指向程序名称),第二个是 NULL

argv[1] 访问 argv 的第二个元素。如果不提供任何参数,argc 将为 1,argv[1] 将为 NULL。因此,您正在取消引用 NULL 指针。

条件 0 != argc 应改为 argc >= 2argc > 1,条件 0 == argc 应为 argc < 2argc <= 1

,

您必须考虑程序的名称,因此您应该改写 if (argc >= 2)。请参阅man

argc 参数的值是命令行参数的数量。 argv 参数是 C 字符串的向量;它的元素是单独的命令行参数字符串。 正在运行的程序的文件名也作为第一个元素包含在向量中; argc 的值计算这个元素。 空指针总是跟在最后一个元素之后:argv[argc] 是这个空指针。

argv[1] 表示索引 1,但在 C 中,您从索引 0 开始。


固定代码

int main(int argc,char **argv)
{
    
    char file_name[30] = "file.txt";
    
    printf("%s",file_name); /* for debugging : doesn't print it */
    
    if (argc >= 2)
        strcpy(file_name,argv[1]);
    else
        return 1;  // exit program with error code.
    WriteToFile(file_name);
    return 0; // exit with success
}