• 主页

  • 投资

  • IT

    🔥
  • 设计

  • 销售

关闭

返回栏目

关闭

返回python栏目

80 - 自带库 - fcntl - 文件锁

作者:

贺及楼

成为作者

更新日期:2024-10-27 18:13:14

fcntl - 文件锁

作用:保证只用1次,一般在多线程,但是只需要1次运行

文件锁fcntl库 简介

Python 的 fcntl 库提供了对文件描述符的控制操作,类似于 Unix 中的 fcntl()ioctl() 系统调用。这个库允许你对文件描述符进行锁定,以实现进程间的同步,以及执行其他文件控制操作。fcntl 库的主要作用包括:

  1. 文件锁定:通过 fcntl.flock()fcntl.lockf() 函数,可以在文件描述符上执行锁定操作,防止其他进程同时修改文件,从而避免数据冲突。这些锁定操作包括排他锁(LOCK_EX)、共享锁(LOCK_SH)以及解锁(LOCK_UN)。

  2. 非阻塞锁定:可以使用 LOCK_NB 标志来执行非阻塞锁定,如果锁定失败,函数会立即返回而不是等待。

  3. 文件控制操作fcntl.fcntl() 函数可以执行各种文件控制操作,如改变文件描述符的属性或获取文件状态标志。

  4. I/O 控制fcntl.ioctl() 函数用于执行 I/O 控制操作,这通常涉及到设备特定的操作,如获取或设置终端属性。

  5. 审计事件:在某些操作中,fcntl 库会触发审计事件,这对于系统安全和监控非常有用。

  6. 文件描述符的非阻塞模式fcntl 库可以设置文件描述符为非阻塞模式,使得文件操作如读取不会阻塞进程,如果没有数据可读,会立即返回错误。

fcntl 库在需要对文件进行进程间同步或执行特定文件控制操作时非常有用,尤其是在 Unix 或类 Unix 系统上进行系统编程时。

文件锁fcntl

  1. import fcntl
  2. import atexit
  3. f = open("scheduler.lock", "wb")
  4. try:
  5. fcntl.flock(f, fcntl.LOCK_EX | fcntl.LOCK_NB)
  6. except:
  7. pass
  8. def unlock():
  9. fcntl.flock(f, fcntl.LOCK_UN)
  10. f.close()
  11. atexit.register(unlock)
  1. flock() : flock(f, operation)

文件锁fcntl 类型

operation参数 锁名 获得者 未获得者
fcntl.LOCK_UN 解锁
fcntl.LOCK_EX 排他锁 读写 没有
fcntl.LOCK_SH 共享锁
fcntl.LOCK_NB 有非阻塞锁 / 立即返回
fcntl.LOCK_NB 无非阻塞锁 /
  1. operation : 包括:
  2. LOCK_SH 共享锁:所有进程没有写访问权限,即使是加锁进程也没有。所有进程有读访问权限。
  3. LOCK_EX 排他锁:除加锁进程外其他进程没有对已加锁文件读写访问权限。
  4. LOCK_NB 非阻塞锁:如果指定此参数,函数不能获得文件锁就立即返回,否则,函数会等待获得文件锁。
  5. LOCK_NB可以同LOCK_SHLOCK_NB进行按位或(|)运算操作。 fcnt.flock(f,fcntl.LOCK_EX|fcntl.LOCK_NB)

文件锁fcntl例子:

  1. import fcntl
  2. import atexit
  3. import multiprocessing
  4. import time
  5. class fcntl_fun():
  6. def __init__(self, lock_file, open_type):
  7. self.lock_file = lock_file
  8. self.file = open(self.lock_file, open_type)
  9. # file = self.open_file()
  10. # def open_file(self):
  11. # self.file = open(self.lock_file, "wb")
  12. # 共享锁_不等
  13. def add_share_no_wait(self):
  14. try:
  15. print("共享锁_不等")
  16. fcntl.flock(self.file, fcntl.LOCK_SH | fcntl.LOCK_NB)
  17. return True
  18. except:
  19. print("共享锁_不等_出错")
  20. return False
  21. # 排他锁_不等
  22. def add_other_no_wait(self):
  23. try:
  24. print("排他锁_不等")
  25. fcntl.flock(self.file, fcntl.LOCK_EX | fcntl.LOCK_NB)
  26. return True
  27. except:
  28. print("排他锁_不等_出错")
  29. return False
  30. # 共享锁_等
  31. def add_share_wait(self):
  32. try:
  33. print("共享锁_等")
  34. fcntl.flock(self.file, fcntl.LOCK_SH)
  35. return True
  36. except:
  37. print("共享锁_等_出错")
  38. return False
  39. # 排他锁_等
  40. def add_other_wait(self):
  41. try:
  42. print("排他锁_等")
  43. fcntl.flock(self.file, fcntl.LOCK_EX)
  44. return True
  45. except:
  46. print("排他锁_等_出错")
  47. return False
  48. # 解锁
  49. def unlock(self):
  50. fcntl.flock(self.file, fcntl.LOCK_UN)
  51. self.file.close()
  52. def write(self, i):
  53. self.file.write(str(i))
  54. self.file.write("\n")
  55. # atexit.register(unlock)
  56. def func_1(number,lock_file, open_type):
  57. print("子进程_1")
  58. f = fcntl_fun(lock_file, open_type)
  59. result = f.add_other_wait() # 换这里
  60. print(result)
  61. if result:
  62. for i in range(number):
  63. time.sleep(1)
  64. print("子进程_1_" + str(i))
  65. def func_2(number,lock_file, open_type):
  66. print("子进程_2")
  67. f = fcntl_fun(lock_file, open_type)
  68. result = f.add_other_wait() # 换这里
  69. print(result)
  70. if result:
  71. for i in range(number):
  72. time.sleep(1)
  73. f.write(i)
  74. print("子进程_2_" + str(i))
  75. if __name__ == "__main__":
  76. begin_time = time.time()
  77. pool = multiprocessing.Pool(processes=2)
  78. # 可以用来换上面验证
  79. # result = f.add_share_no_wait() # 共享锁_不等
  80. # result = f.add_other_no_wait() # 排他锁_不等
  81. # result = f.add_share_wait() # 共享锁_等
  82. # result = f.add_other_wait() # 排他锁_等
  83. # print(result)
  84. # lock_file = "scheduler.lock"
  85. lock_file = "/Users/chu/Desktop/scheduler.txt"
  86. open_type = "w"
  87. res = pool.apply_async(func_1, (10,lock_file, open_type))
  88. time.sleep(3)
  89. res = pool.apply_async(func_2, (10,lock_file, open_type))
  90. pool.close()
  91. pool.join()
  92. end_time = time.time()
  93. print(end_time - begin_time)

文件锁fcntl 总结

共享锁_不等 排他锁_不等 共享锁_等 排他锁_等
W 子进程_1 子进程_1 子进程_1 子进程_1
共享锁_不等 排他锁_不等 共享锁_等 排他锁_等
TRUE TRUE TRUE TRUE
子进程_1_0 子进程_1_0 子进程_1_0 子进程_1_0
子进程_1_1 子进程_1_1 子进程_1_1 子进程_1_1
子进程_1_2 子进程_1_2 子进程_1_2 子进程_1_2
子进程_2 子进程_2 子进程_2 子进程_2
共享锁_不等 排他锁_不等 共享锁_等 排他锁_等
TRUE 排他锁不等出错 TRUE 子进程_1_3
子进程_1_3 FALSE 子进程_1_3 子进程_1_4
子进程_2_0 子进程_1_3 子进程_2_0 子进程_1_5
子进程_1_4 子进程_1_4 子进程_1_4 子进程_1_6
子进程_2_1 子进程_1_5 子进程_2_1 子进程_1_7
子进程_1_5 子进程_1_6 子进程_1_5 子进程_1_8
子进程_2_2 子进程_1_7 子进程_2_2 子进程_1_9
子进程_1_6 子进程_1_8 子进程_1_6 TRUE
子进程_2_3 子进程_1_9 子进程_2_3 子进程_2_0
子进程_1_7 子进程_1_7 子进程_2_1
子进程_2_4 子进程_2_4 子进程_2_2
子进程_1_8 子进程_1_8 子进程_2_3
子进程_2_5 子进程_2_5 子进程_2_4
子进程_1_9 子进程_1_9 子进程_2_5
子进程_2_6 子进程_2_6 子进程_2_6
子进程_2_7 子进程_2_7 子进程_2_7
子进程_2_8 子进程_2_8 子进程_2_8
子进程_2_9 子进程_2_9 子进程_2_9
时间 13s 10s 13s 20s