前言
假如要你写一段代码创建一个文件( 如果文件已经存在则返回失败 )。你会怎么做?
本文将讨论两种做法,并进行分析。
错误代码示例
1 if ((fd = open(pathname, O_WRONLY)) < 0) { 2 if (errno == ENOENT) { 3 if ((fd = creat(pathname, mode)) < 0) 4 err_sys("creat error"); 5 } 6 else { 7 err_sys("open error"); 8 } 9 }
错误分析
这段代码错在没有考虑到操作系统中的进程并发。设想,如果有另一个进程在open和creat函数之间也creat了这个文件,那么当本段代码执行到第三行,就会有两个进程同时对文件进行处理,必然导致混乱。
解决办法
我们可以将open和creat函数之间的处理设为一个原子操作,即这段代码要么不执行,一旦执行就必须执行完。
open函数带上 O_CREAT | O_EXCL 参数后,能以原子操作的方式实现这个功能。
正确代码示例
1 if ((fd = open(pathname, O_CREAT|O_EXCL)) < 0) { 2 err_sys("open error"); 3 }
说明
在系统编程中,我们要多多考虑并发可能给程序带来的影响。