让类/进程/脚本「单身」的方法

简介: 有某些场景下,我们不希望有多个相同的 Linux 进程 或 Shell 脚本同时执行,因为相同进程同时执行,可能会破坏数据的一致性。当然还有在 C++ 代码里,有时希望保证程序中一个类只有一个实例,并提供一个访问它的全局访问点,也就是所谓的「单例模式」。只有一个实例很重要,比如一个打印机可以有多个打印任务,但是只有一个正在工作的任务,一个系统只能有一个窗口管理器或文件系统。 

前言


有某些场景下,我们不希望有多个相同的 Linux 进程 或 Shell 脚本同时执行,因为相同进程同时执行,可能会破坏数据的一致性

当然还有在 C++ 代码里,有时希望保证程序中一个类只有一个实例,并提供一个访问它的全局访问点,也就是所谓的「单例模式」。只有一个实例很重要,比如一个打印机可以有多个打印任务,但是只有一个正在工作的任务,一个系统只能有一个窗口管理器或文件系统。 

接下来,简单介绍下:

  • Linux 命令的方式控制进程是「单例」的方式;
  • C 代码单进程控制的实现;
  • C++ 线程安全的「单例模式」实现。


正文


flock 命令为脚本加锁


可以用flock命令为 Shell 脚本加锁。当多个进程可能会执行同一个脚本,这些进程需要保证其它进程没有在操作,以免重复执行。通常,这样的进程会使用一个「锁文件」,也就是建立一个文件来告诉别的进程自己在运行,如果检测到那个文件存在则认为有操作同样数据的进程在工作

flock命令来为脚本加锁,如下命令:


flock -xn <锁文件> -c <shell脚本>


  • -x : 获取一个排它锁,或者称为写入锁,为默认项
  • -n : 非阻塞模式,当获取锁失败时,返回 1 而不是等待
  • -c : 执行命令或脚本


实战演示


1. 编写一个测试脚本 test.sh


#! /bin/bash
echo "Hello World"
sleep 1000


2. flock 命令给脚本加锁


flock -xn ./test.lock -c "/root/test.sh"


9.png


3. 开启另外一个 bash 窗口运行同个的脚本


10.png


另外一个 bash 窗口运行了同个脚本后,未获取到锁直接返回了,直到上一个脚本运行完毕,这个才可以开始正常运行。

应用的场景

可以在 Linux 定时器/etc/crontab里运用flock命令为脚本加锁,防止重复执行:


* * * * * (flock -xn ./test.lock -c "/root/test.sh")


C 代码实现单进程控制


通常后台服务器程序都必须有且只有一个进程,那么如何控制单进程呢?思想和上面提到的flock命令差不多。

我们可以通过flock系统接口函数对某个文件进行加锁

  • 若加锁不正常,说明后台服务进程已经在运行了,这时则直接报错退出;
  • 若加锁成功,说明后台服务进程没有在运行,这时可以正常启用进程。


用 flock 函数实现的单进程控制代码


12.jpg

C 程序单进程控制


实战演练


我们在 main 函数使用上面的函数:


int main(void)
{
    //进程单实例运行检测
    if(0 != server_is_running())
    {
        printf("myserver process is running!!!!! Current process will exit !\n");
        return -1;
    }
    while(1)
    {
        printf("myserver doing ... \n");
        sleep(2);
    }
    return 0;
}


运行程序,可知进程pid是 6965


[root@lincoding singleprocess]# ./myserver 
server is not running! begin to run..... pid=6965
myserver doing ... 
myserver doing ... 


此时,再运行同个程序,这时会报错退出,因为检测到程序已经在运行中,不可以起另外一个进程。


[root@lincoding singleprocess]# ./myserver 
server is runing now! errno=11
myserver process is running!!!!! Current process will exit !


C++ 单例模式


单例模式指在整个系统生命周期里,保证一个类只能产生一个实例,确保该类的唯一性

单例类的特点:

  • 声明「构造函数和析构函数」为 private 类型,目的禁止外部构造和析构
  • 声明「复制构造和赋值操作」函数为 private 类型,目的是禁止外部拷贝和赋值,确保实例的唯一性
  • 类里有个获取实例的「静态函数」,可以全局访问

还有需要注意的是写单例类时,要注意多线程的竞争的问题,因为可能存在当两个线程同时获取单例对象时,产生出了两个对象,这就违背了单例模式的唯一性。

单例模式实现的方式有很多种,这里推荐一下相对比较简洁的懒汉式单例的两种写法:


在 C++ 11 标准中提出「局部静态变量」初始化具有线程安全性,那么此时写出一个线程安全的单例类,只需要几行代码。


13.jpg

局部静态对象单例模式实现


Single 使用的静态变量是一个「局部静态变量」,因此只有在 Single 的GetInstance()函数被调用时其才会被创建,从而拥有了延迟初始化(Lazy)的效果,提高了程序的启动性能。同时该实例将生存至程序执行完毕。而就 Single 的用户代码而言,其生存期贯穿于整个程序生命周期,从程序启动开始直到程序执行完。


同时,C++ 11 也提供一个新的东西叫std::call_once,配合std::once_flag,可以保证函数在任何情况下只调用一次。


14.jpg

std::call_once 单例模式实现


小结


15.png

相关文章
|
4月前
|
运维 Linux
Linux查找占用的端口,并杀死进程的简单方法
通过上述步骤和命令,您能够迅速识别并根据实际情况管理Linux系统中占用特定端口的进程。为了获得更全面的服务器管理技巧和解决方案,提供了丰富的资源和专业服务,是您提升运维技能的理想选择。
185 1
|
9月前
|
Linux
【Linux】命名管道的创建方法&&基于命名管道的两个进程通信的实现
【Linux】命名管道的创建方法&&基于命名管道的两个进程通信的实现
127 0
|
5月前
|
存储 监控
【Azure Cloud Service】在Azure云服务中收集CPU监控指标和IIS进程的DUMP方法
在使用Cloud Service服务时,发现服务的CPU占用很高,在业务请求并不大的情况下,需要直到到底是什么进程占用了大量的CPU资源,已经如何获取IIS进程(w3wp.exe)的DUMP文件?
|
9月前
|
安全 Linux 开发者
⭐⭐⭐⭐⭐Linux C/C++ 进程崩溃诊断以及有效数据收集:解锁代码问题快速定位与修复的方法
⭐⭐⭐⭐⭐Linux C/C++ 进程崩溃诊断以及有效数据收集:解锁代码问题快速定位与修复的方法
497 1
|
5月前
|
监控 Ubuntu API
Python脚本监控Ubuntu系统进程内存的实现方式
通过这种方法,我们可以很容易地监控Ubuntu系统中进程的内存使用情况,对于性能分析和资源管理具有很大的帮助。这只是 `psutil`库功能的冰山一角,`psutil`还能够提供更多关于系统和进程的详细信息,强烈推荐进一步探索这个强大的库。
79 1
|
5月前
|
编译器
【收藏】内核级利用通用Hook函数方法检测进程
【收藏】内核级利用通用Hook函数方法检测进程
|
6月前
|
数据安全/隐私保护 异构计算 Windows
【Azure 环境】 介绍两种常规的方法来监视Window系统的CPU高时的进程信息: Performance Monitor 和 Powershell Get-Counter
【Azure 环境】 介绍两种常规的方法来监视Window系统的CPU高时的进程信息: Performance Monitor 和 Powershell Get-Counter
|
6月前
|
Linux Perl
在Linux中,系统目前有许多正在运行的任务,在不重启机器的条件下,有什么方法可以把所有正在运行的进程移除呢?
在Linux中,系统目前有许多正在运行的任务,在不重启机器的条件下,有什么方法可以把所有正在运行的进程移除呢?
|
7月前
|
运维 Python Windows
如何通过Python脚本查找并终止占用指定端口的进程
在日常的开发和运维过程中,某些端口被意外占用是一个常见的问题。这种情况可能导致服务无法启动或冲突。本文将介绍如何通过Python脚本查找并终止占用指定端口的进程,以确保系统的正常运行。
|
6月前
|
机器学习/深度学习 数据可视化 搜索推荐
低代码开发是一种能够加速软件研发进程的高效开发方法
【8月更文挑战第4天】低代码开发是一种能够加速软件研发进程的高效开发方法
71 0

热门文章

最新文章

相关实验场景

更多