新闻  |   论坛  |   博客  |   在线研讨会
fopen自动创建文件失败(文件路径检查创建)
电子禅石 | 2020-04-14 12:41:55    阅读:2479   发布文章

一、问题描述
      fopen函数使用过程发现,当fopen要打开的文件路径中,包含一个不存在路径时,不管fopen函数的第二参数设成什么,都无法建立该文件。实例:char *filename = "/usr/local/ct/mirror_im_protocol.cfg";FILE *fp = NULL; if ((fp=fopen(filename,"a"))==NULL){    printf("open file %s error\n",filename);    return FAIL; }
      上述实例中,若/usr/local/ct/不存在,fopen将不会建立mirror_im_protocol.cfg文件,导致fopen返回NULL。当/usr/local/ct/路径存在,而mirror_im_protocol.cfg不存在时,函数会自动创建该文件,并往文件中追加内容。
      因此,在使用fopen函数创建一个新文件时,一定要保证文件的路径存在,可以考虑在fopen函数前加一个路径检查。
二、路径检查
      access函数可以用来判断文件夹或者文件是否存在。
函数原型: int access(const char *filename, int mode);
所属头文件:io.h
filename:可以填写文件夹路径或者文件路径
mode:0 (F_OK) 只判断是否存在
               2 (R_OK) 判断写入权限
               4 (W_OK) 判断读取权限
               6 (X_OK) 判断执行权限
      用于判断文件夹是否存在的时候,mode取0,判断文件是否存在的时候,mode可以取0、2、4、6。 若存在或者具有权限,返回值为0;不存在或者无权限,返回值为-1。
错误代码
EACCESS 参数pathname 所指定的文件不符合所要求测试的权限。
EROFS 欲测试写入权限的文件存在于只读文件系统内。
EFAULT 参数pathname指针超出可存取内存空间。
EINVAL 参数mode 不正确。
ENAMETOOLONG 参数pathname太长。
ENOTDIR 参数pathname为一目录。
ENOMEM 核心内存不足
ELOOP 参数pathname有过多符号连接问题。
EIO I/O 存取错误。
特别提醒:使用access()作用户认证方面的判断要特别小心,例如在access()后再做open()的空文件可能会造成系统安全上的问题。
 实例:
#include <stdio.h> #include <io.h> #define  AUDI_PROCESS_PID_PATH  " /usr/local/ct/" int main(void) {     if ( access(AUDI_PROCESS_PID_PATH, F_OK) == 0) )          printf("AUDI_PROCESS_PID_PATH EXISITS!");     else          printf("AUDI_PROCESS_PID_PATH DOESN'T EXISIT!");     return 0; }       当检查到路径不存在时,可通过下面的函数来进行路径的创建。三、创建目录原型:int mkdir (const char *filename, mode_t mode)
返回0表示成功,返回-1表述出错。
使用该函数需要包含头文件sys/stat.h
mode 表示新目录的权限,可以取以下值:
S_IRUSR
S_IREAD
Read permission bit for the owner of the file. On many systems thisbit is 0400. S_IREAD is an obsolete synonym provided for BSDcompatibility.
//这是对文件用户的读权限赋予
S_IWUSR
S_IWRITE
Write permission bit for the owner of the file. Usually 0200.S_IWRITE is an obsolete synonym provided for BSDcompatibility.
//这是对文件用户的写权限赋予
S_IXUSR
S_IEXEC
Execute (for ordinary files) or search (for directories) permissionbit for the owner of the file. Usually 0100. S_IEXEC is an obsoletesynonym provided for BSD compatibility.
//这是对文件用户执行权限赋予
S_IRWXU
This is equivalent to (S_IRUSR | S_IWUSR | S_IXUSR).
//这包含了以上三种用户权限
S_IRGRP
Read permission bit for the group owner of the file. Usually040.
//这是对文件用户组读权限赋予
S_IWGRP
Write permission bit for the group owner of the file. Usually020.
//这是对文件用户组写权限赋予
S_IXGRP
Execute or search permission bit for the group owner of the file.Usually 010.
//这是对文件用户组执行权限赋予
S_IRWXG
This is equivalent to (S_IRGRP | S_IWGRP | S_IXGRP).
//这包含了以上三种组用户权限
S_IROTH
Read permission bit for other users. Usually 04.
//对其他用户的读操作赋予
S_IWOTH
Write permission bit for other users. Usually 02.
//对其他用户的写操作赋予
S_IXOTH
Execute or search permission bit for other users. Usually 01.
//对其他用户的执行操作赋予
S_IRWXO
This is equivalent to (S_IROTH | S_IWOTH | S_IXOTH).
//这包含了以上三种组其他用户权限赋予
S_ISUID
This is the set-user-ID on execute bit, usually 04000. See HowChange Persona.
//这下面就不说明了,因为这又涉及set uid ,set gid,stickybit这三个权限,后面有一篇转贴再说明,不过这也只有我这样的菜鸟不知道的啦.
S_ISGID
This is the set-group-ID on execute bit, usually 02000. See HowChange Persona.
S_ISVTX实例:#include <sys/types.h>#include <sys/stat.h>  int main(){    int status;     status = mkdir("/home/newdir", S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH);      return 0;}这样就创建了一个newdir目录,权限通过ls -al 查看为
drwxr-xr-x 跟用linux命令mkdir创建的目录权限位一致。
四、fopen函数
函数原型:FILE * fopen(const char * path,const char * mode);

返回值:文件顺利打开后,指向该流的文件指针就会被返回。如果文件打开失败则返回NULL,并把错误代码存在errno中。

一般而言,打开文件后会做一些文件读取或写入的动作,若打开文件失败,接下来的读写动作也无法顺利进行,所以一般在fopen()后作错误判断及处理。

参数说明:

参数path字符串包含欲打开的文件路径及文件名,参数mode字符串则代表着流形态。

mode有下列几种形态字符串:

“r” 以只读方式打开文件,该文件必须存在。

“r+” 以可读写方式打开文件,该文件必须存在。

”rb+“ 读写打开一个二进制文件,允许读写数据,文件必须存在。

“w” 打开只写文件,若文件存在则文件长度清为0,即该文件内容会消失。若文件不存在则建立该文件。

“w+” 打开可读写文件,若文件存在则文件长度清为零,即该文件内容会消失。若文件不存在则建立该文件。

“a” 以附加的方式打开只写文件。若文件不存在,则会建立该文件,如果文件存在,写入的数据会被加到文件尾,即文件原先的内容会被保留。(EOF符保留)

”a+“ 以附加方式打开可读写的文件。若文件不存在,则会建立该文件,如果文件存在,写入的数据会被加到文件尾后,即文件原先的内容会被保留。 (原来的EOF符不保留)

“wb” 只写打开或新建一个二进制文件;只允许写数据。

“wb+” 读写打开或建立一个二进制文件,允许读和写

“wx” 创建文本文件,只允许写入数据.[C11]

“wbx” 创建一个二进制文件,只允许写入数据.[C11]

”w+x“ 创建一个文本文件,允许读写.[C11]

“wb+x” 创建一个二进制文件,允许读写.[C11]

“w+bx” 和"wb+x"相同[C11]
————————————————

原文链接:https://blog.csdn.net/u013485792/article/details/52327718


*博客内容为网友个人发布,仅代表博主个人观点,如有侵权请联系工作人员删除。

参与讨论
登录后参与讨论
属于自己的技术积累分享,成为嵌入式系统研发高手。
推荐文章
最近访客