即使我尝试用C语言创建文件,也找不到文件时出现分段错误

即使我尝试用C语言创建文件,也找不到文件时出现分段错误,c,file,segmentation-fault,fopen,argv,C,File,Segmentation Fault,Fopen,Argv,我正在编写一个代码,将文本附加到文件中。 它在WriteToFile的开头使用fopen,因此即使文件不存在,它也会创建它 但是,当我根本不输入任何文件时会发生什么?完全没有争论?只是/a.out 分段错误 我不知道为什么,在我看来,我做了每件事都很好,以避免任何问题 int main(int argc, char **argv) { char file_name[30] = "file.txt"; printf("%s"

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

但是,当我根本不输入任何文件时会发生什么?完全没有争论?只是
/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数组中):

对于这两种情况,我得到

分段故障(堆芯转储)
如果(0!=argc)
argc
值通常是(a)包括程序名在内的参数计数。因此,运行
/a.out
将使
argc
为1而不是零。而且,由于
argv[argc]
通常为
NULL
,因此取消对它的引用不会很好地结束:-)

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

if(argc>1){//同时具有argv[0]和argv[1]。
nowSafeToUse(argv[1]);
}
更详细地说,C11标准规定:

如果已声明,则主功能的参数应遵守以下约束:

  • argc
    的值应为非负
  • argv[argc]
    应为空指针
  • 如果
    argc
    的值大于零,则通过
    argv[argc-1]
    的数组成员
    argv[0]
    应包含指向字符串的指针,这些字符串在程序启动之前由主机环境给定实现定义的值。目的是向程序提供在程序启动之前从托管环境中的其他位置确定的信息。如果主机环境无法提供包含大写和小写字母的字符串,则实现应确保以小写字母接收字符串
  • 如果
    argc
    的值大于零,则
    argv[0]
    指向的字符串表示程序名<代码>argv[0][0]如果程序名在主机环境中不可用,则应为空字符。如果
    argc
    的值大于1,则
    argv[1]
    通过
    argv[argc-1]
    指向的字符串表示程序参数
  • 程序应可修改参数
    argc
    argv
    以及
    argv
    数组指向的字符串,并在程序启动和程序终止之间保留其最后存储的值

关于这一行,另请注意:

printf(“%s”,文件名);/*用于调试:不打印它*/
如果这不是打印,可能是因为标准输出是行缓冲的(如果确定为交互式设备,则为默认值,否则为完全缓冲)

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

printf(“%s”,文件名);
puts(文件名);

另一方面,如果您输入的文件名超过29个字符,您将遇到麻烦,因为它将溢出
文件名
,并允许在末尾使用
\0

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

int main(int argc,char**argv){
char*文件名=(argv>1)
?argv[1]
:“file.txt”;
printf(“%s\n”,文件名);//用于调试:可能会打印它:-)
WriteToFile(文件名);
}

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

。。。这些值是在程序启动之前由主机环境给定的实现定义的值。目的是向程序提供在程序启动之前从托管环境中的其他位置确定的信息

几乎意味着它可以做任何它想做的事情:-)

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

如果(0!=argc)
argc
值通常是(a)包括程序名在内的参数计数。因此,运行
/a.out
将使
argc
为1而不是零。而且,由于
argv[argc]
通常为
NULL
,因此取消对它的引用不会很好地结束:-)

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

if(argc>1){//同时具有argv[0]和argv[1]。
nowSafeToUse(argv[1]);
}
更详细地说,C11标准规定:

如果已声明,则主功能的参数应遵守以下约束:

  • argc
    的值应为非负
  • argv[argc]
    应为空指针
  • 如果
    argc
    的值大于零,则通过
    argv[argc-1]
    的数组成员
    argv[0]
    应包含指向字符串的指针,这些字符串在程序启动之前由主机环境给定实现定义的值。目的是向程序提供在程序启动之前从托管环境中的其他位置确定的信息。如果主机环境无法提供包含大写和小写字母的字符串,则实现应确保以小写字母接收字符串
  • 如果
    argc
    的值大于零,则
    argv[0]
    指向的字符串表示程序名<如果程序名不是av,则代码>argv[0][0]应为空字符
    char file_name[30];
        
        if (0 == argc)
        {
           strcpy(file_name, "file.txt");
        }
        else
        {
            strcpy(file_name, argv[1]);
        }