当前位置:首页 > 嵌入式 > 嵌入式教程
[导读]在多任务操作系统环境下,多个进程会同时运行,并且一些进程之间可能存在一定的关联。多个进程可能为了完成同一个任务会相互协作,这样形成进程之间的同步关系。而且在不同进程之间,为了争夺有限的系统资源(硬件或软件资源)会进入竞争状态,这就是进程之间的互斥关系。

8.4信号量8.4.1信号量概述

在多任务操作系统环境下,多个进程会同时运行,并且一些进程之间可能存在一定的关联。多个进程可能为了完成同一个任务会相互协作,这样形成进程之间的同步关系。而且在不同进程之间,为了争夺有限的系统资源(硬件或软件资源)会进入竞争状态,这就是进程之间的互斥关系。

进程之间的互斥与同步关系存在的根源在于临界资源。临界资源是在同一个时刻只允许有限个(通常只有一个)进程可以访问(读)或修改(写)的资源,通常包括硬件资源(处理器、内存、存储器以及其他外围设备等)和软件资源(共享代码段,共享结构和变量等)。访问临界资源的代码叫做临界区,临界区本身也会成为临界资源。

信号量是用来解决进程之间的同步与互斥问题的一种进程之间通信机制,包括一个称为信号量的变量和在该信号量下等待资源的进程等待队列,以及对信号量进行的两个原子操作(PV操作)。其中信号量对应于某一种资源,取一个非负的整型值。信号量值指的是当前可用的该资源的数量,若它等于0则意味着目前没有可用的资源。PV原子操作的具体定义如下:

P操作:如果有可用的资源(信号量值>0),则占用一个资源(给信号量值减去一,进入临界区代码);如果没有可用的资源(信号量值等于0),则被阻塞到,直到系统将资源分配给该进程(进入等待队列,一直等到资源轮到该进程)。

V操作:如果在该信号量的等待队列中有进程在等待资源,则唤醒一个阻塞进程。如果没有进程等待它,则释放一个资源(给信号量值加一)。

使用信号量访问临界区的伪代码所下所示:

{

/*设R为某种资源,S为资源R的信号量*/

INIT_VAL(S);/*对信号量S进行初始化*/

非临界区;

P(S);/*进行P操作*/

临界区(使用资源R);/*只有有限个(通常只有一个)进程被允许进入该区*/

V(S);/*进行V操作*/

非临界区;

}

最简单的信号量是只能取0和1两种值,这种信号量被叫做二维信号量。在本节中,主要讨论二维信号量。二维信号量的应用比较容易地扩展到使用多维信号量的情况。

8.4.2信号量的应用1.函数说明

在Linux系统中,使用信号量通常分为以下几个步骤。

(1)创建信号量或获得在系统已存在的信号量,此时需要调用semget()函数。不同进程通过使用同一个信号量键值来获得同一个信号量。

(2)初始化信号量,此时使用semctl()函数的SETVAL操作。当使用二维信号量时,通常将信号量初始化为1。

(3)进行信号量的PV操作,此时调用semop()函数。这一步是实现进程之间的同步和互斥的核心工作部分。

(4)如果不需要信号量,则从系统中删除它,此时使用semclt()函数的IPC_RMID操作。此时需要注意,在程序中不应该出现对已经被删除的信号量的操作。

2.函数格式

表8.17列举了semget()函数的语法要点。

表8.17 semget()函数语法要点

所需头文件

#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/sem.h>

函数原型

intsemget(key_tkey,intnsems,intsemflg)

函数传入值

key:信号量的键值,多个进程可以通过它访问同一个信号量,其中有个特殊值IPC_PRIVATE。它用于创建当前进程的私有信号量

nsems:需要创建的信号量数目,通常取值为1

semflg:同open()函数的权限位,也可以用八进制表示法,其中使用IPC_CREAT标志创建新的信号量,即使该信号量已经存在(具有同一个键值的信号量已在系统中存在),也不会出错。如果同时使用IPC_EXCL标志可以创建一个新的唯一的信号量,此时如果该信号量已经存在,该函数会返回出错

函数返回值

成功:信号量标识符,在信号量的其他函数中都会使用该值

出错:-1

表8.18列举了semctl()函数的语法要点。

表8.18 semctl()函数语法要点

所需头文件

#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/sem.h>

函数原型

intsemctl(intsemid,intsemnum,intcmd,unionsemunarg)

函数传入值

semid:semget()函数返回的信号量标识符

semnum:信号量编号,当使用信号量集时才会被用到。通常取值为0,就是使用单个信号量(也是第一个信号量)

cmd:指定对信号量的各种操作,当使用单个信号量(而不是信号量集)时,常用的有以下几种:

IPC_STAT:获得该信号量(或者信号量集合)的semid_ds结构,并存放在由第4个参数arg的buf指向的semid_ds结构中。semid_ds是在系统中描述信号量的数据结构。

IPC_SETVAL:将信号量值设置为arg的val值

IPC_GETVAL:返回信号量的当前值

IPC_RMID:从系统中,删除信号量(或者信号量集)

arg:是unionsemnn结构,该结构可能在某些系统中并不给出定义,此时必须由程序员自己定义

unionsemun

{

intval;

structsemid_ds*buf;

unsignedshort*array;

}

函数返回值

成功:根据cmd值的不同而返回不同的值

IPC_STAT、IPC_SETVAL、IPC_RMID:返回0

IPC_GETVAL:返回信号量的当前值

出错:-1

表8.19列举了semop()函数的语法要点。

表8.19 semop()函数语法要点

所需头文件

#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/sem.h>

函数原型

intsemop(intsemid,structsembuf*sops,size_tnsops)

函数传入值

semid:semget()函数返回的信号量标识符

sops:指向信号量操作数组,一个数组包括以下成员:

structsembuf

{

shortsem_num;/*信号量编号,使用单个信号量时,通常取值为0*/

shortsem_op;

/*信号量操作:取值为-1则表示P操作,取值为+1则表示V操作*/

shortsem_flg;

/*通常设置为SEM_UNDO。这样在进程没释放信号量而退出时,系统自动

释放该进程中未释放的信号量*/

}

nsops:操作数组sops中的操作个数(元素数目),通常取值为1(一个操作)

函数返回值

成功:信号量标识符,在信号量的其他函数中都会使用该值

出错:-1

3.使用实例

本实例说明信号量的概念以及基本用法。在实例程序中,首先创建一个子进程,接下来使用信号量来控制两个进程(父子进程)之间的执行顺序。

因为信号量相关的函数调用接口比较复杂,我们可以将它们封装成二维单个信号量的几个基本函数。它们分别为信号量初始化函数(或者信号量赋值函数)init_sem()、P操作函数sem_p()、V操作函数sem_v()以及删除信号量的函数del_sem()等,具体实现如下所示:

/*sem_com.c*/

#include"sem_com.h"

/*信号量初始化(赋值)函数*/

intinit_sem(intsem_id,intinit_value)

{

unionsemunsem_union;

sem_union.val=init_value;/*init_value为初始值*/

if(semctl(sem_id,0,SETVAL,sem_union)==-1)

{

perror("Initializesemaphore");

return-1;

}

return0;

}

/*从系统中删除信号量的函数*/

intdel_sem(intsem_id)

{

unionsemunsem_union;

if(semctl(sem_id,0,IPC_RMID,sem_union)==-1)

{

perror("Deletesemaphore");

return-1;

}

}

/*P操作函数*/

intsem_p(intsem_id)

{

structsembufsem_b;

sem_b.sem_num=0;/*单个信号量的编号应该为0*/

sem_b.sem_op=-1;/*表示P操作*/

sem_b.sem_flg=SEM_UNDO;/*系统自动释放将会在系统中残留的信号量*/

if(semop(sem_id,&sem_b,1)==-1)

{

perror("Poperation");

return-1;

}

return0;

}

/*V操作函数*/

intsem_v(intsem_id)

{

structsembufsem_b;

sem_b.sem_num=0;/*单个信号量的编号应该为0*/

sem_b.sem_op=1;/*表示V操作*/

sem_b.sem_flg=SEM_UNDO;/*系统自动释放将会在系统中残留的信号量*/

if(semop(sem_id,&sem_b,1)==-1)

{

perror("Voperation");

return-1;

}

return0;

}

现在我们调用这些简单易用的接口,可以轻松解决控制两个进程之间的执行顺序的同步问题。实现代码如下所示:

/*fork.c*/

#include<sys/types.h>

#include<unistd.h>

#include<stdio.h>

#include<stdlib.h>

#include<sys/types.h>

#include<sys/ipc.h>

#include<sys/shm.h>

#defineDELAY_TIME3/*为了突出演示效果,等待几秒钟,*/

intmain(void)

{

pid_tresult;

intsem_id;

sem_id=semget(ftok(".",'a'),1,0666|IPC_CREAT);/*创建一个信号量*/

init_sem(sem_id,0);

/*调用fork()函数*/

result=fork();

if(result==-1)

{

perror("Fork\n");

}

elseif(result==0)/*返回值为0代表子进程*/

{

printf("Childprocesswillwaitforsomeseconds...\n");

sleep(DELAY_TIME);

printf("Thereturnedvalueis%dinthechildprocess(PID=%d)\n",

result,getpid());

sem_v(sem_id);

}

else/*返回值大于0代表父进程*/

{

sem_p(sem_id);

printf("Thereturnedvalueis%dinthefatherprocess(PID=%d)\n",

result,getpid());

sem_v(sem_id);

del_sem(sem_id);

}

exit(0);

}

读者可以先从该程序中删除掉信号量相关的代码部分并观察运行结果。

$./simple_fork

Childprocesswillwaitforsomeseconds…/*子进程在运行中*/

Thereturnedvalueis4185inthefatherprocess(PID=4184)/*父进程先结束*/

[…]$Thereturnedvalueis0inthechildprocess(PID=4185)/*子进程后结束了*/

再添加信号量的控制部分并运行结果。

$./sem_fork

Childprocesswillwaitforsomeseconds…

/*子进程在运行中,父进程在等待子进程结束*/

Thereturnedvalueis0inthechildprocess(PID=4185)/*子进程结束了*/

Thereturnedvalueis4185inthefatherprocess(PID=4184)/*父进程结束*/

本实例说明使用信号量怎么解决多进程之间存在的同步问题。我们将在后面讲述的共享内存和消息队列的实例中,看到使用信号量实现多进程之间的互斥。

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

9月2日消息,不造车的华为或将催生出更大的独角兽公司,随着阿维塔和赛力斯的入局,华为引望愈发显得引人瞩目。

关键字: 阿维塔 塞力斯 华为

加利福尼亚州圣克拉拉县2024年8月30日 /美通社/ -- 数字化转型技术解决方案公司Trianz今天宣布,该公司与Amazon Web Services (AWS)签订了...

关键字: AWS AN BSP 数字化

伦敦2024年8月29日 /美通社/ -- 英国汽车技术公司SODA.Auto推出其旗舰产品SODA V,这是全球首款涵盖汽车工程师从创意到认证的所有需求的工具,可用于创建软件定义汽车。 SODA V工具的开发耗时1.5...

关键字: 汽车 人工智能 智能驱动 BSP

北京2024年8月28日 /美通社/ -- 越来越多用户希望企业业务能7×24不间断运行,同时企业却面临越来越多业务中断的风险,如企业系统复杂性的增加,频繁的功能更新和发布等。如何确保业务连续性,提升韧性,成...

关键字: 亚马逊 解密 控制平面 BSP

8月30日消息,据媒体报道,腾讯和网易近期正在缩减他们对日本游戏市场的投资。

关键字: 腾讯 编码器 CPU

8月28日消息,今天上午,2024中国国际大数据产业博览会开幕式在贵阳举行,华为董事、质量流程IT总裁陶景文发表了演讲。

关键字: 华为 12nm EDA 半导体

8月28日消息,在2024中国国际大数据产业博览会上,华为常务董事、华为云CEO张平安发表演讲称,数字世界的话语权最终是由生态的繁荣决定的。

关键字: 华为 12nm 手机 卫星通信

要点: 有效应对环境变化,经营业绩稳中有升 落实提质增效举措,毛利润率延续升势 战略布局成效显著,战新业务引领增长 以科技创新为引领,提升企业核心竞争力 坚持高质量发展策略,塑强核心竞争优势...

关键字: 通信 BSP 电信运营商 数字经济

北京2024年8月27日 /美通社/ -- 8月21日,由中央广播电视总台与中国电影电视技术学会联合牵头组建的NVI技术创新联盟在BIRTV2024超高清全产业链发展研讨会上宣布正式成立。 活动现场 NVI技术创新联...

关键字: VI 传输协议 音频 BSP

北京2024年8月27日 /美通社/ -- 在8月23日举办的2024年长三角生态绿色一体化发展示范区联合招商会上,软通动力信息技术(集团)股份有限公司(以下简称"软通动力")与长三角投资(上海)有限...

关键字: BSP 信息技术
关闭
关闭