概述:


最近在分析一个偶现的问题,偶现概率特别低,问题还在分析中。把分析的知识做个总结,后面再继续补充。


问题描述:


代码在调用Lua的require函数时发生异常,通过查看require的源码跟踪,发现该函数的fopen函数返回打开文件失败的异常,

下面就来总结下fopen打开文件出错可能有哪些原因,也许不全,欢迎补充。


函数介绍:


函数原型 FILE * fopen(const char * path,const char * mode); 

函数功能 打开一个文件

参数:   path [in] 名称   mode[in] 打开方式

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

注:这里只是简单介绍下函数功能和参数,具体参数意思此处不分析。


原因分析及方法


1 参数path问题, 路径不对fopen就会是返回失败。

分析:首先查看path文件是否存在,其次检查path的路径相对路径还是绝对路径?如果是相对路径再检查是否当前进程的目录会切换了,软件找不到文件了.

如 相对路径 FILE *fp = fopen("./test/1.txt", "r"); 

  绝对路径 FILE *fp = fopen("/mnt/text/1.txt", "r");

假如确定参数path没问题后,则可以排除文件不存在的路径问题。此时可以打印错误码errno来定位问题(这个后面介绍)。


2 参数mode问题,mode控制文件打开的方式,如果用户打开的方式超出了当前用户的权限,那么fopen也会返回失败,

此时应该检查当前用户的操作权限,也可以打印错误码errno来定位问题

如果当前用户仅仅只有读的权限而以读写的方式打开文件 FILE *fp = fopen("./test/1.txt", "W+")


3 检查程序中是否有句柄泄露的可能即频繁的调用fopen而没有fclose,这种情况的表象就是前面刚刚开始的时候可以open成功

过一段时间后,怎么都open不成功了,检查路径和权限都没有问题, 那此时就要检查下是否句柄泄露了。一般linux最多支持1000来个

句柄,打开太多不关,则其他的没法打开了


4 通过检查errno来分析定位问题, errno是一个int型的值,在errno.h中定义不需要自己定义。

可以通过strerror(errno)查看错误信息, errno是调试程序的一个重要方法。


注:errno 是记录系统的最后一次错误代码。

例如 

FILE *fp = NULL;
if ((fp =fopen("./test/1.txt", "r")) == NULL)
{
printf("open fail errno = %d reason = %s \n", errno, strerrno(errno));
}


需要指出的是这样加入printf出问题,那么分局errno是记录系统最后一次错误代码,则有可能得不到我们想要的错误码,反而误导
最好的办法就是

FILE *fp = NULL;
int errNum = 0;
if ((fp =fopen("./test/1.txt", "r")) == NULL)
{
errNum = errno;
printf("open fail errno = %d reason = %s \n", errNum, strerrno(errNum));
}



常见的errno错误码有以下这些:

#define EPERM 1 /* Operation not permitted */ 
   #define ENOENT 2 /* No such file or directory */
   #define ESRCH 3 /* No such process */
   #define EIO 5 /* I/O error */
   #define ENXIO 6 /* No such device or address */
   #define E2BIG 7 /* Argument list too long */
   #define ENOEXEC 8 /* Exec format error */
   #define EBADF 9 /* Bad file number */
   #define ECHILD 10 /* No child processes */




更多的请网上搜索, 这里就分析这些,后续有发现继续补充...