fopen函数失败原因分析以及分析原因的方法

程序发现 fopen 函数失败了,但找不到原因,在网上搜了下,找到一篇文章,发现了问题所在。

因为写的是测试程序,没有注意 fopen 后没有及时 fclose 。所以一开始的时候,程序都没有报错,最后打开了几千个文件句柄才发现一直打不开文件了。

使用 ulimit -n 可以查看当前系统中打开文件句柄的最大值,我这设置的是 4096。(对每个用户而言)

使用 lsof 命令可以查看,第一列是 command; 第二列是 pid; 第三列是 user; lsof 命令可以参考:lsof命令

可以使用以下命令统计各进程打开句柄数 (第一列是打开文件句柄数,第二列是pid),以打开文件句柄数降序排列

lsof - n|awk ’ {print $2}’| sort | uniq -c | sort -nr | more

在后面再 grep 一下我的进程 pid,就可以找到我的进程有多少打开文件句柄(其实第一个就是了- -)

问题的根本还是在于 fopen 一定要 fclose。为什么我会遗漏这个 ,是因为这个程序里 fopen 是在一个线程,操作文件内容是在别的线程。所以在结束的时候,遗忘 fclose。

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

问题描述:
代码在调用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 */

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


作者:奋斗Andy
来源:CSDN
原文:https://blog.csdn.net/Andy_93/article/details/78865000
版权声明:本文为博主原创文章,转载请附上博文链接!

转自《fopen函数失败原因分析以及分析原因的方法》

参考 《Linux下查看进程打开的文件句柄数》

  • 1
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值