91-互斥量的鲁棒属性

2024-03-23 14:20
文章标签 互斥 属性 91 鲁棒

本文主要是介绍91-互斥量的鲁棒属性,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

上一文有一个遗留练习,很重要,不知道同学们有没有做。先说结论吧,如果其中一个进程在未释放互斥量的情况下挂掉了,将会导致另一个线程永远无法获得锁,然后就死锁了。

为了能够让进程在异常终止时,释放掉互斥锁,需要指定 ROBUST 属性。所谓的 ROBUST,指是的健壮的意思。

1. 相关函数

可以通过下面两个函数设置和获取互斥量的鲁棒属性:

int pthread_mutexattr_getrobust(const pthread_mutexattr_t *attr, int *restrict robust);int pthread_mutexattr_setrobust(pthread_mutexattr_t *attr, int robust);

2. 互斥量状态一致性

在指定 robust 属性的情况下,如果其中某个进程在未释放锁的情况下退出了,另一个进程仍然可以获得锁,但是此时 pthread_mutex_lock 将返回 EOWNERDEAD,通知获得锁的线程,有一个其它进程的线程挂掉了,互斥量现在变成了 inconsistent 的状态。这时候,需要对互斥量做 consistent 处理,否则,一旦再次解锁后,互斥量将永久不可用。

翻译成代码就是这样的:

if (EOWNERDEAD == pthread_mutex_lock(&lock)) {pthread_mutex_consistent(&lock);
}

consistent 函数原型如下:

int pthread_mutex_consistent(pthread_mutex_t *mutex);

它表示将 robust mutex 标记为 consistent 状态。

3. 程序清单

我们仍使用上一篇文章中的 init、destroy 程序,下面给出 rbstbuyticket.c 的代码。

3.1 代码

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <pthread.h>#define PERR(msg) do { perror(msg); exit(-1); } while(0)
#define PPERR(err, msg) do { err = errno; perror(msg); exit(-1); } while(0)struct ticket {int remain;pthread_mutex_t lock;
};int main(int argc, char* argv[]) {if (argc < 2) {printf("Usage: %s <name>\n", argv[0]);exit(-1);}char *name = argv[1];int err, shared, flag = 1;key_t key = 0x8888;int id = shmget(key, 0, 0); if (id < 0) PERR("shmget");struct ticket *t = (struct ticket*)shmat(id, NULL, 0); if ((int)t == -1) PERR("shmat");while(flag) {if (EOWNERDEAD == (err = pthread_mutex_lock(&t->lock))) {puts("EOWNERDEAD");pthread_mutex_consistent(&t->lock);}   else if (ENOTRECOVERABLE == err) {puts("ENOTRECOVERABLE");}   int remain = t->remain;if (remain > 0) {sleep(1);printf("%s buy a ticket\n", name);--remain;sleep(3);t->remain = remain;}   else flag = 0;pthread_mutex_unlock(&t->lock);sleep(2);}return 0;
}

3.2 Makefile 文件

添加了两行,现在如下:

main:init destroy buyticket rbstbuyticket 
init:init.cgcc init.c -o init -lpthread
destroy:destroy.cgcc destroy.c -o destroy -lpthread
buyticket:buyticket.cgcc buyticket.c -o buyticket -lpthread
rbstbuyticket:rbstbuyticket.cgcc rbstbuyticket.c -o rbstbuyticket -lpthread

3.3 编译和运行

  • 编译
$ make
  • 未指定 robust 下的运行结果

这里写图片描述

图1 未指定 robust 导致死锁
  • 指定 robust 的运行结果(init 带参数)

这里写图片描述

图2 指定 robust 后正常

细心的同学发现,allen 抢了两张票,而 luffy 抢了 4 张票!!! 不过这个问题似乎是没办法解决……如果你有解决方案,请在评论区留言。

4. 总结

  • 理解 robust 属性
  • 注意对 mutex 做一致性处理

这篇关于91-互斥量的鲁棒属性的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/838564

相关文章

Linux线程之线程的创建、属性、回收、退出、取消方式

《Linux线程之线程的创建、属性、回收、退出、取消方式》文章总结了线程管理核心知识:线程号唯一、创建方式、属性设置(如分离状态与栈大小)、回收机制(join/detach)、退出方法(返回/pthr... 目录1. 线程号2. 线程的创建3. 线程属性4. 线程的回收5. 线程的退出6. 线程的取消7.

python删除xml中的w:ascii属性的步骤

《python删除xml中的w:ascii属性的步骤》使用xml.etree.ElementTree删除WordXML中w:ascii属性,需注册命名空间并定位rFonts元素,通过del操作删除属... 可以使用python的XML.etree.ElementTree模块通过以下步骤删除XML中的w:as

Python打印对象所有属性和值的方法小结

《Python打印对象所有属性和值的方法小结》在Python开发过程中,调试代码时经常需要查看对象的当前状态,也就是对象的所有属性和对应的值,然而,Python并没有像PHP的print_r那样直接提... 目录python中打印对象所有属性和值的方法实现步骤1. 使用vars()和pprint()2. 使

CSS3中的字体及相关属性详解

《CSS3中的字体及相关属性详解》:本文主要介绍了CSS3中的字体及相关属性,详细内容请阅读本文,希望能对你有所帮助... 字体网页字体的三个来源:用户机器上安装的字体,放心使用。保存在第三方网站上的字体,例如Typekit和Google,可以link标签链接到你的页面上。保存在你自己Web服务器上的字

SpringBoot读取ZooKeeper(ZK)属性的方法实现

《SpringBoot读取ZooKeeper(ZK)属性的方法实现》本文主要介绍了SpringBoot读取ZooKeeper(ZK)属性的方法实现,强调使用@ConfigurationProperti... 目录1. 在配置文件中定义 ZK 属性application.propertiesapplicati

Java反射实现多属性去重与分组功能

《Java反射实现多属性去重与分组功能》在Java开发中,​​List是一种非常常用的数据结构,通常我们会遇到这样的问题:如何处理​​List​​​中的相同字段?无论是去重还是分组,合理的操作可以提高... 目录一、开发环境与基础组件准备1.环境配置:2. 代码结构说明:二、基础反射工具:BeanUtils

MySQL 事务的概念及ACID属性和使用详解

《MySQL事务的概念及ACID属性和使用详解》MySQL通过多线程实现存储工作,因此在并发访问场景中,事务确保了数据操作的一致性和可靠性,下面通过本文给大家介绍MySQL事务的概念及ACID属性和... 目录一、什么是事务二、事务的属性及使用2.1 事务的 ACID 属性2.2 为什么存在事务2.3 事务

Spring Cache注解@Cacheable的九个属性详解

《SpringCache注解@Cacheable的九个属性详解》在@Cacheable注解的使用中,共有9个属性供我们来使用,这9个属性分别是:value、cacheNames、key、key... 目录1.value/cacheNames 属性2.key属性3.keyGeneratjavascriptor

Spring Boot 事务详解(事务传播行为、事务属性)

《SpringBoot事务详解(事务传播行为、事务属性)》SpringBoot提供了强大的事务管理功能,通过@Transactional注解可以方便地配置事务的传播行为和属性,本文将详细介绍Spr... 目录Spring Boot 事务详解引言声明式事务管理示例编程式事务管理示例事务传播行为1. REQUI

Spring Security基于数据库的ABAC属性权限模型实战开发教程

《SpringSecurity基于数据库的ABAC属性权限模型实战开发教程》:本文主要介绍SpringSecurity基于数据库的ABAC属性权限模型实战开发教程,本文给大家介绍的非常详细,对大... 目录1. 前言2. 权限决策依据RBACABAC综合对比3. 数据库表结构说明4. 实战开始5. MyBA