readdir的原型如下:

struct dirent *readdir(DIR *dirp);

因为内部使用了静态数据,所以readdir被认为不是线程安全的函数,POSIX[i]标准这样描述:

The application shall not modify the structure to which the return value of readdir() points, nor any storage areas pointed to by pointers within the structure. The returned
pointer, and pointers within the structure, might be invalidated or the structure or the storage areas might be overwritten by a subsequent call to readdir() on the same directory stream. They shall not be affected by a call to readdir()
on a different directory stream.

If a file is removed from or added to the directory after the most recent call to opendir() or rewinddir(),whether a subsequent call to readdir() returns
an entry for that file is unspecified.

The readdir() function need not be thread-safe.

因此才有了readdir_r函数的出现,它的原型如下:

int readdir_r(DIR *dirp, struct dirent *entry, struct dirent **result);

readdir_r将返回结果填充到调用者提供的entry缓冲区中,保证了它的线程安全性。

然而,在GNU的官方文档[ii]中,有下面的描述:

In POSIX.1-2008, readdir is not thread-safe. In the GNU C Library implementation, it is safe to call readdir concurrently on different dirstreams, but multiple threads accessing the
same dirstream result in undefined behavior. readdir_r is a fully thread-safe alternative, but suffers from poor portability (see below).
It is recommended that you use readdir, with externallocking if multiple threads access the same dirstream.

Portability Note:
It is recommended to use readdir instead of readdir_r for the following reasons:

On systems which do not define NAME_MAX, it may not be possible to use readdir_r safely because the caller does not specify the length of the buffer for the directory entry.

On some systems,readdir_r cannot read directory entries with very long names. If such a name is encountered, the GNU C Library implementation of readdir_r returns with an error code
of ENAMETOOLONG after the final directory entry has been read. On other systems, readdir_r may return successfully, but the d_name member may not be NUL-terminated or may be truncated.

POSIX-1.2008 does not guarantee that readdir is thread-safe, even when access to the same dirstream is serialized.
But in current implementations(including the GNU C Library), it is safe to call readdir concurrently on different dirstreams, so there is no need to use readdir_r in most multi-threaded programs.
In the rare case that multiple threads need to read from the same dirstream, it is still better to use readdir and external synchronization.

It is expected that future versions of POSIX will obsolete readdir_r and mandate the level of thread safety for readdir which is provided by the GNU C Library and other implementations
today.

尽管POSIX中不保证readdir是线程安全的,但是在目前的实现中(包括GUN  C库),在不同的dirstream上(dirp)同时调用readdir能够保证是安全的。因此,多线程程序中其实没必要使用readdir_r,即使在极少场景下,多个线程中需要使用相同的dirstream时,使用readdir以及外部同步手段(加锁),也会是更好的选择。预计在未来版本的POSIX标准中,将会废弃readdir_r。

除了线程安全方面的考虑,没必要使用readdir_r之外,readdir_r还有其他可移植上的缺点,比如某些系统上readdir_r无法处理有很长名字的目录项。

而且,结构体dirent中只有d_name是在POSIX中有明确规定的,它长度还是未指定的(在Linux中,结构体dirent中的d_name,具有明确的数组长度,为256),因此,某些系统中,为了使用readdir_r,必须像下面这样分配entry的内存:

name_max = pathconf(dirpath, _PC_NAME_MAX);
if (name_max == -1) /* Limit not defined, or error */
name_max = 255; /* Take a guess */
len = offsetof(struct dirent, d_name) + name_max + 1;
entryp = malloc(len);

但是这种方式也有问题[iii]

因此,结论就是:

只要不是多个线程使用相同的dirstream,就尽可能的使用readdir,它其实更简单且更安全。

 

参考:

http://pubs.opengroup.org/onlinepubs/9699919799/

https://www.gnu.org/software/libc/manual/html_mono/libc.html#Reading_002fClosing-Directory

http://elliotth.blogspot.co.uk/2012/10/how-not-to-use-readdirr3.html

 


[i] http://pubs.opengroup.org/onlinepubs/9699919799/

[ii] https://www.gnu.org/software/libc/manual/html_mono/libc.html#Reading_002fClosing-Directory

[iii] elliotth's blog: How (not) to usereaddir_r(3):http://elliotth.blogspot.co.uk/2012/10/how-not-to-use-readdirr3.html

是readdir,还是readdir_r的更多相关文章

  1. C++ readdir、readdir_r函数

    readdir, readdir_r - 读一个目录 readdir函数: struct dirent *readdir(DIR *dirp); The  data  returned by read ...

  2. Thinkphp拖拽上传文件-使用webuploader插件(自己改动了一些地方)——分片上传

    html页面: <!DOCTYPE html> <html class="js cssanimations"> <head> <meta  ...

  3. Linux c readdir是非线程安全,需用readdir_r,要注意用静态变量当做返回值的函数的非线程安全性

    readdir函数: struct dirent *readdir(DIR *dirp); The  data  returned by readdir() may be overwritten by ...

  4. readdir_r()读取目录内容

    readdir()在多线程操作中不安全,Linux提供了readdir_r()实现多线程读取目录内容操作. #include <stdio.h> #include <stdlib.h ...

  5. PHP之readdir()函数

    最近在学习php文件操作的相关知识,记录一下readdir()函数其中的一个要注意的点 1. 在$temp=readdir($handle)函数中 readdir获取的是文件名和$handle中的文件 ...

  6. 关于readdir返回值中struct dirent.d_type的取值有关问题(转)

    关于readdir返回值中struct dirent.d_type的取值问题 原网页链接 http://www.gnu.org/software/libc/manual/html_node/Direc ...

  7. 目录操作函数opendir、readdir和closedir

    首先,明确一个类型DIR的含义: #include <dirent.h> DIR    A type representing a directory stream. DIR是在目录项格式 ...

  8. (转)PHP自定义遍历目录下所有文件dir(),readdir()函数

    方法一:使用dir()遍历目录 dir()函数,成功时返回Directory类实例 PHP dir() 语法格式为: dir(directory);//directory为需要显示文件名的目录名称,可 ...

  9. oendir(),readdir(),closedir() 打开/读取/关闭目录

    目录操作 当目标是目录而不是文件的时候,ls -l的结果会显示目录下所有子条目的信息,怎么去遍历整个目录呢?答案马上揭晓! 1. 打开目录 功能:opendir()用来打开参数name指定的目录,并返 ...

随机推荐

  1. nginx的四个基本功能

    Nginx能做什么 1.反向代理2.负载均衡3.HTTP服务器(包含动静分离)4.正向代理 以上就是做网站小编了解到的Nginx在不依赖第三方模块能处理的事情,下面详细说明每种功能怎么做 1.反向代理 ...

  2. Python学习(二) 基础语法之初看python

    Python 标识符 略 Python保留字符 一大堆,说了未必记得住,编码过程中慢慢去记住. 行和缩进 这个要说一下,学习Python与其他语言最大的区别就是,Python的代码块不使用大括号({} ...

  3. hystrix熔断器

    在一个具有多服务的应用中,假如由于其中某一个服务出现问题,导致响应速度变慢,或是根本没有响应返回,会导致它的服务消费者由于长时间的等待,消耗尽线程,进而影响到对其他服务的线程调用,进而会转变为整个应用 ...

  4. oracle习题练习-表空间-用户-表-约束

    题一 1.       创建名字为hy_tablespace的表空间,默认大小为10M;@@ 2.       创建一个用户,用户名以自己名字命名,并指定命名空间为hy_tablespace;@@@@ ...

  5. Python实例 分割路径和文件名

    import  os.path # 常用函数有三种:分隔路径,找出文件名.找出盘符(windows系统),找出文件的扩展名. # 根据你机器的实际情况修改下面参数. spath = " D: ...

  6. UnicodeEncodeError:'latin-1' codec can't encode characters in position 0-1: ordinal not in range(256) Scrapy

    1.使用scrapy对数据进行入库时,出现如下错误: UnicodeEncodeError:'latin-1' codec can't encode characters in position 0- ...

  7. Vue.js NPM 安装方法

    由于 npm 安装速度慢,本教程使用了淘宝的镜像及其命令 cnpm,安装使用介绍参照:使用淘宝 NPM 镜像. npm 版本需要大于 3.0,如果低于此版本需要升级它: # 查看版本 $ npm -v ...

  8. LintCode_372 在O(1)时间复杂度删除链表节点

    题目 给定一个单链表中的表头和一个等待被删除的节点(非表头或表尾).请在在O(1)时间复杂度删除该链表节点.并在删除该节点后,返回表头. 思路 因为O(1)内删除所以 不能从头节点去遍历找他的前驱节点 ...

  9. FZU 1576【计算几何/费马点】

    Oaiei居住在A城市,并且是这个城市建设的总设计师.最近有个问题一直困恼着他.A城市里有三个大型工厂,每个大型工厂每天都需要消耗大量的石油,现在城市里要建设一个石油中转站,从石油中转站到三个大型工厂 ...

  10. Python 运算符括号