Linux系统中信号量的基本使用方法

简介: Linux系统中信号量的基本使用方法

    信号量(Semaphore),有时被称为信号灯,是在多线程环境下使用的一种设施,是可以用来保证两个或多个关键代码段不被并发调用。在进入一个关键代码段之前,线程必须获取一个信号量;一旦该关键代码段完成了,那么该线程必须释放信号量。其它想进入该关键代码段的线程必须等待直到第一个线程释放信号量。

第一: 父子进程相隔1s报数一次

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <string.h>
#include <sys/sem.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h>
typedef union semun {
               int              val;    /* Value for SETVAL */
               unsigned short  *array;  /* Array for GETALL, SETALL */
}SEM;
//实现两个进程的0~99报数
int main(int argc,char *argv[])
{
  int i=0;
  //1、获取键值 ftok
  key_t key = ftok(argv[1], 2);
  if(-1 == key)
  {
    perror("ftok error\r\n");
    return 0;
  }
  //2、创建信号量集 semget
  int semid = semget(key, 2, IPC_CREAT | 0666);
  if(-1 == semid)
  {
    perror("semget error\r\n");
    return 0;
  }
  SEM num={0};
  //3、设置0号信号量的值为1 SETVAL
  num.val = 1;
  if(semctl(semid, 0, SETVAL, num) == -1)
  {
    perror("SETALL error\r\n");
    return 0;
  }
  //4、创建操作方式的结构体 struct sembuf
  struct sembuf sops[4]={{0,1,0},{0,-1,0},{1,1,0},{1,-1,0}};
  //5、创建子进程 fork
  pid_t pid = fork( );
  if(pid < 0)
  {
    perror("fork error\r\n");
    return 0;
  }else if(0 == pid)//子进程 单数
  {
     i=1;  //初始为1
    while(i < 100)
    {
      //信号1    -1
      if(semop(semid, &sops[3], 1) == -1)
      {
        perror("semop 1 -1 error\r\n");
        return 0;
      }
      printf("child:time = %d\r\n",i);// 1 
      i+=2;
      sleep(1);
      //信号0    +1
      if(semop(semid, &sops[0], 1) == -1)
      {
        perror("semop 0 +1 error\r\n");
        return 0;
      }
    }
    exit(0);
  }else//父进程 双数
  {
    while(i < 100)
    {
      //信号0    -1
      if(semop(semid, &sops[1], 1) == -1)
      {
        perror("semop 0 -1 error\r\n");
        return 0;
      }
      printf("father:time = %d\r\n",i);  //0 2
      i+=2;
            sleep(1);
      //信号1    +1
      if(semop(semid, &sops[2], 1) == -1)
      {
        perror("semop 1 +1 error\r\n");
        return 0;
      }
    }
    wait(NULL);   
  }
  return 0;
}

第二:利用信号量实现父子进程通信

//实例2:利用父子进程通信
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/mman.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <sys/wait.h>
#include <sys/sem.h>
union semun
{
  int val;
  struct semid_ds *buf;
  unsigned short *array;
  struct seminfo* _buf; 
};
int main()
{
  int ret;
  int semid;
  //创建信号量
  semid = semget(IPC_PRIVATE,1,IPC_CREAT|0644);
  //初始化信号量、设置信号量
  union semun arg;
  arg.val=0;//信号量值为0
  semctl(semid,0,SETALL,arg);
  //查询当前信号量的值
  ret = semctl(semid,0,GETVAL);//GETVAL返回值为信号值
  printf("val:%d\r\n",ret);
  //创建子进程
  pid_t pid;
  pid = fork();
  if(pid == 0)
  {
    //子进程
    sleep(5);
    struct sembuf sops={0,1,0};
    while(1)
    {
      semop(semid,&sops,1);//P操作
      printf("--------V------\r\n");
      sleep(1);
    }
  }
  else
  {
    //父进程
    struct sembuf sops={0,-1,0};
    while(1)
    {
      semop(semid,&sops,1);//P操作
      printf("--------P------\r\n");//信号量0还可以执行最后一次
      sleep(1);
    }
  }
  semctl(semid,0,IPC_RMID);
  return 0;
}

总结:信号量就是为了对共享资源进行保护,理解基本实现方法非常重要。

目录
相关文章
|
8月前
|
Ubuntu Linux Anolis
Linux系统禁用swap
本文介绍了在新版本Linux系统(如Ubuntu 20.04+、CentOS Stream、openEuler等)中禁用swap的两种方法。传统通过注释/etc/fstab中swap行的方式已失效,现需使用systemd管理swap.target服务或在/etc/fstab中添加noauto参数实现禁用。方法1通过屏蔽swap.target适用于新版系统,方法2通过修改fstab挂载选项更通用,兼容所有系统。
754 3
Linux系统禁用swap
|
8月前
|
Linux
Linux系统修改网卡名为eth0、eth1
在Linux系统中,可通过修改GRUB配置和创建Udev规则或使用systemd链接文件,将网卡名改为`eth0`、`eth1`等传统命名方式,适用于多种发行版并支持多网卡配置。
1325 3
|
Ubuntu Linux 网络安全
Linux系统初始化脚本
一款支持Rocky、CentOS、Ubuntu、Debian、openEuler等主流Linux发行版的系统初始化Shell脚本,涵盖网络配置、主机名设置、镜像源更换、安全加固等多项功能,适配单/双网卡环境,支持UEFI引导,提供多版本下载与持续更新。
831 3
Linux系统初始化脚本
|
8月前
|
安全 Linux Shell
Linux系统提权方式全面总结:从基础到高级攻防技术
本文全面总结Linux系统提权技术,涵盖权限体系、配置错误、漏洞利用、密码攻击等方法,帮助安全研究人员掌握攻防技术,提升系统防护能力。
1009 1
|
8月前
|
监控 安全 Linux
Linux系统提权之计划任务(Cron Jobs)提权
在Linux系统中,计划任务(Cron Jobs)常用于定时执行脚本或命令。若配置不当,攻击者可利用其提权至root权限。常见漏洞包括可写的Cron脚本、目录、通配符注入及PATH变量劫持。攻击者通过修改脚本、创建恶意任务或注入命令实现提权。系统管理员应遵循最小权限原则、使用绝对路径、避免通配符、设置安全PATH并定期审计,以防范此类攻击。
1320 1
|
8月前
|
Linux 应用服务中间件 Shell
二、Linux文本处理与文件操作核心命令
熟悉了Linux的基本“行走”后,就该拿起真正的“工具”干活了。用grep这个“放大镜”在文件里搜索内容,用find这个“探测器”在系统中寻找文件,再用tar把东西打包带走。最关键的是要学会使用管道符|,它像一条流水线,能把这些命令串联起来,让简单工具组合出强大的功能,比如 ps -ef | grep 'nginx' 就能快速找出nginx进程。
963 1
二、Linux文本处理与文件操作核心命令
|
8月前
|
Linux
linux命令—stat
`stat` 是 Linux 系统中用于查看文件或文件系统详细状态信息的命令。相比 `ls -l`,它提供更全面的信息,包括文件大小、权限、所有者、时间戳(最后访问、修改、状态变更时间)、inode 号、设备信息等。其常用选项包括 `-f` 查看文件系统状态、`-t` 以简洁格式输出、`-L` 跟踪符号链接,以及 `-c` 或 `--format` 自定义输出格式。通过这些选项,用户可以灵活获取所需信息,适用于系统调试、权限检查、磁盘管理等场景。
549 137
|
8月前
|
安全 Ubuntu Unix
一、初识 Linux 与基本命令
玩转Linux命令行,就像探索一座新城市。首先要熟悉它的“地图”,也就是/根目录下/etc(放配置)、/home(住家)这些核心区域。然后掌握几个“生存口令”:用ls看周围,cd去别处,mkdir建新房,cp/mv搬东西,再用cat或tail看文件内容。最后,别忘了随时按Tab键,它能帮你自动补全命令和路径,是提高效率的第一神器。
1463 58
|
7月前
|
存储 安全 Linux
Linux卡在emergency mode怎么办?xfs_repair 命令轻松解决
Linux虚拟机遇紧急模式?别慌!多因磁盘挂载失败。本文教你通过日志定位问题,用`xfs_repair`等工具修复文件系统,三步快速恢复。掌握查日志、修磁盘、验重启,轻松应对紧急模式,保障系统稳定运行。
1331 2