Linux编程--文件原子 *** 作

Linux编程--文件原子 *** 作,第1张

当多个进程同时访问一个文件的时候,普通的write/read在执行的时候,无法保证 *** 作原子性,可能会导致文件被污染,达不到预期的结果。

任何一个需要多个函数调用的 *** 作都不可能是原子 *** 作,因为在两个函数调用间,内核可能会将进程挂起执行另外的进程。

如果想要避免这种情况的话,则需要使用pread/pwrite函数

ssize_t pread(int fd ,void *buffer ,size_t size,off_t offset)

返回真正读取到的字节数,offset是指的从文件开始位置起的offset个字节数开始读。其余的参数与read无异。

PS:

pread是无法中断的原子 *** 作,无法中断它的定位和读取 *** 作

pread读取过后的文件偏移量不会发生改变

同理pwrite也是一样的

而在文件创建的时候也是一样的,当需要做文件创建同步的时候,我们需要在O_CREATE的时候,加上O_EXCL标志位,当已经创建过的话,会返回fd,否则返回错误

int dup( int filedes):

传入一个文件描述符,返回当前可用的最小文件描述符。

int dup2(int filedes,int filedes2):

传入文件描述符,以及新的文件描述符,如果新的文件描述符所指向的文件已经打开,则会强行将其关闭后,将该文件描述符指向到已存在的文件描述符。

如果filedes和filedes2指向同一个文件,则不做任何处理,直接返回filedes2,不会关闭文件

新返回回来的filedes2会共享filedes的文件状态标识,文件偏移量等等信息。因为它们的文件指针会指向文件表的同一个位置。只是fd不一样而已。

Linux的源码中x86体系结构原子 *** 作的定义文件为。

linux2.6/include/asm-i386/atomic.h

文件内定义了原子类型atomic_t,其仅有一个字段counter,用于保存32位的数据。

typedef struct { volatile int counter} atomic_t

其中原子 *** 作函数atomic_inc完成自加原子 *** 作。

/**

* atomic_inc - increment atomic variable

* @v: pointer of type atomic_t

*

* Atomically increments @v by 1.

*/

static __inline__ void atomic_inc(atomic_t *v)

{

__asm__ __volatile__(

LOCK "incl %0"

:"=m" (v->counter)

:"m" (v->counter))

}

其中LOCK宏的定义为。

#ifdef CONFIG_SMP

#define LOCK "lock "

#else

#define LOCK ""

#endif

可见,在对称多处理器架构的情况下,LOCK被解释为指令前缀lock。而对于单处理器架构,LOCK不包含任何内容。


欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/yw/7449542.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2023-04-05
下一篇2023-04-05

发表评论

登录后才能评论

评论列表(0条)

    保存