write调用的原子性

unix环境高级编程的3.11节讲述了原子操作,其中有一种情形是在文件尾端添加数据。文中说,如果多

个进程都需要将数据添加到某一文件,那么为了保证定位和写数据这两步是一个原子操作,需要在打开

文件时设置O_APPEND标志,看到这里我们就会想,虽然保证了定位和写数据是一个原子操作,但是是否

能够保证多个进程或线程写入的数据不会交错呢,比如A进程调用write(filedes1, "AAA", 3),B进程调

用write(filedes2, "BBBB", 4)(其中filedes1和filedes2指向同一个文件),但是最后文件中的数据

是否有可能是AABBBAB,如果这个文件是一个管道或socket呢。linux man手册页中关于write调用的说明

很不详细,并未说明写操作是否是原子的,所以我们有必要查找Single UNIX Specification(SUS)对

write调用的说明,在SUS中对此调用的说明还是比较详细的。在继续讨论之前我们需要清楚 内核在写文

件之前会对该文件加锁,不管是否成功完成写操作,在返回之前都会解锁。下面我们就以三种常见的文

件根据SUS标准来讨论上面提出的这个问题:

1.普通文件

SUS中也没有说明在写普通文件时是否会保证是原子操作,但是它说明了write调用可能并不能完全把我

们需要写入的数据写到文件中去,那么什么情况下可能少写数据呢?SUS说明了两种情况: 磁盘已满或则

要写入的文件的大小超过了当前进程的文件大小限制。其实至少还有一种情况,那就是内核中的高速缓

存不够用的时候,比如linux内核在发现高速缓存不够用的时候就只写入实际能够容下的数据然后返回。

正是由于存在上述最后一种情况,所以说按照APUE那种方法在linux下面写文件并不能保证我们的数据不

会交错(不过我们可以根据write的返回值得知是否有发生交错的可能)。其它的unix内核可能会在实现上

不同于linux内核,他们可能在写之前就判断一下缓冲区是否足够容纳所有数据,如果是这种情况,写操

作应该就是原子的;也可能写了一部分数据后才发现缓冲区不够用并让当前进程进入睡眠状态,此时内

核如果解锁,那么在当前进程睡眠期间其它进程可能写了数据,如果不解锁,那么就是原子操作,其他

进程不可能在这个时候写入数据。由上面的分析可知,正是由于SUS标准不太完整的标准,我们不能确定

一定可以按APUE的方法来同时向同一个普通文件写数据。如果我们非要在同一个文件中记录多个进程产

生的数据,我们最好采用unix日志系统采用的方法,用一个专用进程处理文件IO,其它进程把需要写的

数据发送给这个专用进程,这样应该比多个进程同时写一个文件可靠和高效。

2.管道

SUS对管道的写操作说得更多也更明确,我们只需遵照其标准就可以了。对于write(pipefd, buf,

nbyte),其要点如下:

如果nbyte <= PIPE_BUF,不管O_NONBLOCK是否设置,其写操作都是原子的,就是说多个进程都在此条件

下同时写同一个管道不会引起数据交错。

如果nbyte > PIPE_BUF,是不能保证写操作是原子的,写入的数据可能与其他进程写入的数据交错。

3.socket

SUS中对于写socket并没有说很多,我们无法从标准中得知write是否保证写操作的原子性。我看了一下

linux 2.6.14内核关于tcp数据的写操作,发现它不是原子的,也从网上查到了这部分代码的作者(们)

对这个问题的看法,他(们)认为对一个可能永久阻塞的操作保证原子性是错误的。我们也只能姑且这

么认为了。
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值