ICode9

精准搜索请尝试: 精确搜索
首页 > 系统相关> 文章详细

linux php 进程进阶(一) 进程锁(文件锁)

2022-01-18 20:33:18  阅读:288  来源: 互联网

标签:fp 文件 进阶 LOCK pid 进程 flock php


进程锁

这里的进程锁与线程锁、互斥锁、读写锁和自旋锁不同,它是通过记录一个PID文件,避免两个进程同时运行的文件锁。

进程锁的作用之一就是可以协调进程的运行,例如crontab使用进程锁解决冲突提到,使用crontab限定每一分钟执行一个任务,但这个进程运行时间可能超过一分钟,如果不用进程锁解决冲突的话两个进程一起执行就会有问题。

通过进程锁可以解决进程间同步的问题。

使用PID文件锁还有一个好处,方便进程向自己发停止或者重启信号。Nginx编译时可指定参数–pid-path=/var/run/nginx.pid,进程起来后就会把当前的PID写入这个文件,当然如果这个文件已经存在了,也就是前一个进程还没有退出,那么Nginx就不会重新启动。进程管理工具Supervisord也是通过记录进程的PID来停止或者拉起它监控的进程的。

使用进程锁

进程锁在特定场景是非常适用的,而操作系统默认不会为每个程序创建进程锁,那我们该如何使用呢?

  1. 其实要实现一个进程锁很简单,通过文件就可以实现了。例如程序开始运行时去检查一个PID文件,如果文件存在就直接退出,如果文件不存在就创建一个,并把当前进程的PID写入文件中。这样我们很容易可以实现进程锁,但是所有流程都需要自己控制。

  2. 当然根据DRY(Don’t Repeat Yourself)原则,Linux已经为我们提供了flock接口。

使用Flock

  1. Flock提供的是advisory lock,也就是建议性的锁,其他进程实际上也可以读写这个锁文件。Linux上可以直接使用flock命令,使用C可以调用原生的flock接口
  2. php 也提供了 flock 文件锁函数,下面我们来实现一下进程锁
$file = 'process.pid';
$fp = fopen($file, 'c');
if (!$fp) {
    throw new \Exception('Can\'t create lock file!');
}
if(flock($fp, LOCK_EX)){
    ftruncate($fp, 0); //清空文件内容
    $pid = getmypid(); //获取当前进程pid
    fwrite($fp, $pid);
    fflush($fp); //释放锁之前刷新输出
    flock($fp , LOCK_UN);//释放锁
}else{
   echo "打不开锁\n";
}

  1. 区分读锁定 和 写 锁定。
    如果每次都使用 写锁定,那么连多个进程读取一个文件也要排队,这样的效率肯定不行。
  2. 读锁:LOCK_SH(share-lock)共享锁,读操作前,期望增加的锁定。导致,允许并发读,阻塞额外的写操作。
  3. 写锁: LOCK_EX(exclusive-lock)排他锁,独占锁,写操作前,尝试添加的锁定类型。导致,其他脚本不能读也不能写。
  4. 这两种都是阻塞模式

阻塞:指的是上一个操作执行完解锁之后,下一个对文件锁定的操作才能继续执行

$file = 'process.pid';
$fp = fopen($file, 'c');
if (!$fp) {
    throw new \Exception('Can\'t create lock file!');
}
if(flock($fp, LOCK_EX)){
    ftruncate($fp, 0); //清空文件内容
    $pid = getmypid(); //获取当前进程pid
    fwrite($fp, $pid);
    fflush($fp); //释放锁之前刷新输出
    sleep(30); //模拟并发
    flock($fp , LOCK_UN);//释放锁
}else{
   echo "打不开锁\n";
}

  1. 可以通过LOCK_NB来在锁定失败时,不阻塞:
$file = 'process.pid';
$fp = fopen($file, 'c');
if (!$fp) {
    throw new \Exception('Can\'t create lock file!');
}
if(flock($fp, LOCK_EX|LOCK_NB)){
    ftruncate($fp, 0); //清空文件内容
    $pid = getmypid(); //获取当前进程pid
    fwrite($fp, $pid);
    fflush($fp); //释放锁之前刷新输出
    sleep(30); //模拟并发
    flock($fp , LOCK_UN);//释放锁
}else{
   echo "打不开锁\n";
}

标签:fp,文件,进阶,LOCK,pid,进程,flock,php
来源: https://blog.csdn.net/csdn_leidada/article/details/122558715

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有