【centos】【python】程序单例

需求:

避免同一个程序在不同时刻、终端并行执行;不是绝对不能,是两个同样的程序尽量不要一块执行;

技术:

在程序执行时将pid写入到一个文件里,结束时再删除pid文件,下一个程序先检查pid文件是否存在再决定是否提前终止;

python 复制代码
import datetime
import shutil
from pathlib import Path
import logging
import os
# import fcntl
import psutil
from logging.handlers import RotatingFileHandler


LOG_PATH_PARENT = "./"
LOG_PATH = LOG_PATH_PARENT + "/log"


logger = logging.getLogger()
logger.setLevel(logging.INFO)

formatter = logging.Formatter('%(asctime)s [%(levelname)s] %(message)s')
log_abs_file = os.path.join(LOG_PATH_PARENT, 'suri_logrotate.log')

handler = RotatingFileHandler(log_abs_file, maxBytes=1024 * 1024 * 10, backupCount=10)  # 大小10MB , 10个
handler.setFormatter(formatter)
logger.addHandler(handler)

console = logging.StreamHandler()
console.setFormatter(formatter)
logger.addHandler(console)



class SingletonProgress:

    def _get_lock(self):
        """开启进程占用锁,确保其他同名称进程不再重复运行"""
        # file_name = os.path.basename(__file__)
        # lock_file_name = f"/var/run/{file_name}.pid"
        # self.fd = open(self.lock_file_name, "w")
        if not os.path.exists(self.lock_file_name):
            with open(self.lock_file_name, "w", encoding='utf-8') as f:
                f.write(str(os.getpid()))
        else:
            if os.path.exists(self.lock_file_name):
                print(self.lock_file_name)
                with open(self.lock_file_name, 'r', encoding='utf8') as f:
                    pid = f.read().strip()
                if not pid:
                    logger.error("Program abnormal exit,no pid, Please try again.")
                    os.remove(self.lock_file_name)
                elif pid and pid.isdigit() and int(pid) not in psutil.pids():
                    logger.error("Program abnormal exit,pid err, Please try again.")
                    os.remove(self.lock_file_name)
                elif self.file_name not in psutil.Process(int(pid)).cmdline():
                    logger.error("Program abnormal exit,pid not, Please try again.")
                    os.remove(self.lock_file_name)
                else:
                    logger.info(f"{self.file_name} have another instance running.")
                    self.remove_lock_file = False

                exit(1)

    def __init__(self):
        self.file_name = os.path.basename(__file__)
        self.lock_file_name = os.path.join(LOG_PATH_PARENT, self.file_name+'.pid')
        self.remove_lock_file = True
        self.err_delete_lock_file_name = False
        self._get_lock()

    def __del__(self):
        # 占用锁的进程被关闭后,释放锁
        if self.remove_lock_file:
            os.remove(self.lock_file_name)


def method():
    import time
    while 1:
        time.sleep(1)
        logger.info(str(os.getpid()))


if __name__ == '__main__':

    keep_lock = SingletonProgress()
    method()
    # del keep_lock
相关推荐
无为之士8 分钟前
Linux自动备份Mysql数据库
linux·数据库·mysql
岑梓铭25 分钟前
(CentOs系统虚拟机)Standalone模式下安装部署“基于Python编写”的Spark框架
linux·python·spark·centos
努力学习的小廉25 分钟前
深入了解Linux —— make和makefile自动化构建工具
linux·服务器·自动化
MZWeiei29 分钟前
Zookeeper基本命令解析
大数据·linux·运维·服务器·zookeeper
游客52039 分钟前
opencv中的各种滤波器简介
图像处理·人工智能·python·opencv·计算机视觉
Eric.Lee202142 分钟前
moviepy将图片序列制作成视频并加载字幕 - python 实现
开发语言·python·音视频·moviepy·字幕视频合成·图像制作为视频
7yewh44 分钟前
嵌入式Linux QT+OpenCV基于人脸识别的考勤系统 项目
linux·开发语言·arm开发·驱动开发·qt·opencv·嵌入式linux
Dontla1 小时前
vscode怎么设置anaconda python解释器(anaconda解释器、vscode解释器)
ide·vscode·python
小张认为的测试1 小时前
Linux性能监控命令_nmon 安装与使用以及生成分析Excel图表
linux·服务器·测试工具·自动化·php·excel·压力测试
打鱼又晒网1 小时前
linux网络套接字 | 深度解析守护进程 | 实现tcp服务守护进程化
linux·网络协议·计算机网络·tcp