操作系统 实验29 同步与互斥

1、并发线程同步与互斥

源程序:

#include <stdio.h>
#include <stdlib.h>
#include <pthread.h>
#include <unistd.h>
#include <string.h>
int num=30,count=10;
pthread_mutex_t mylock=PTHREAD_MUTEX_INITIALIZER; 
void *sub1(void *arg) {int i = 0,tmp;for (; i <count; i++){pthread_mutex_lock(&mylock);    			tmp=num-1;usleep(13);num=tmp;								pthread_mutex_unlock(&mylock);	        		printf("线程1 num减1后值为: %d\n",num);}return ((void *)0);
}
void *sub2(void *arg){int i=0,tmp;for(;i<count;i++){pthread_mutex_lock(&mylock);tmp=num-1;usleep(31);num=tmp;pthread_mutex_unlock(&mylock);printf("线程2 num减1后值为: %d\n",num);}return ((void *)0);
}
int main(int argc, char** argv) {pthread_t tid1,tid2;int err,i=0,tmp;void *tret;err=pthread_create(&tid1,NULL,sub1,NULL);if(err!=0){printf("pthread_create error:%s\n",strerror(err));exit(-1);}
err=pthread_create(&tid2,NULL,sub2,NULL);if(err!=0){printf("pthread_create error:%s\n",strerror(err));exit(-1);}for(;i<count;i++){pthread_mutex_lock(&mylock);     			tmp=num-1;usleep(5);	num=tmp;		pthread_mutex_unlock(&mylock); printf("main num减1后值为: %d\n",num);}printf("两个线程运行结束\n");err=pthread_join(tid1,&tret);
if(err!=0){printf("can not join with thread1:%s\n",strerror(err));exit(-1);}printf("thread 1 exit code %d\n",(int)tret);err=pthread_join(tid2,&tret);if(err!=0){printf("can not join with thread1:%s\n",strerror(err));exit(-1);}printf("thread 2 exit code %d\n",(int)tret);return 0;
}

编译链接命令:gcc threadmutex.c -o threadmutex -lpthread

运行命令:./threadmutex

交互与结果:

2、生产者-消费者同步与互斥试验

源程序:

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <pthread.h>
#include <unistd.h>
#include <signal.h>
#include <semaphore.h>
#define Maxbuf 10		//缓冲单元数目
#define TimesOfOp 10		//生产者、消费者循环读写缓冲区的次数
#define true 1
#define false 0
#define historynum 100	//生产者、消费者读写历史记录数目
struct Circlebuf	//循环缓冲队列结构
{int read; 			//读指针int write; 		//写指针int buf[Maxbuf]; 	//缓冲区
} circlebuf;
sem_t mutex;			//互斥信号量
sem_t empty; 			//空白缓冲区同步信号量
sem_t full; 			//满缓冲区同步信号量
char writehistory[historynum][30];//写历史
char readhistory[historynum][30];//读历史
int writehistorycount=0;		//写历史计数器
int readhistorycount=0; 		//读历史计数器
char history[historynum][30];	//缓冲区操作历史
int historycount=0;		//缓冲区操作历史计数器
void writeCirclebuf(struct Circlebuf *circlebuf,int *value) //向缓冲区中写一个值
{circlebuf->buf[circlebuf->write]=(*value);sleep(1);							circlebuf->write=(circlebuf->write+1)%Maxbuf; 
}
int readCirclebuf(struct Circlebuf *circlebuf)
{int value=0;value=circlebuf->buf[circlebuf->read];		sleep(1);								circlebuf->buf[circlebuf->read]=0; 			circlebuf->read=(circlebuf->read+1)%Maxbuf; 	return value;
}
void sigend(int sig)
{exit(0);
}
void * productThread(void *i)
{int *n=(int *)i;int t=TimesOfOp;int writeptr;while(t--){	sem_wait(&empty);sem_wait(&mutex);writeCirclebuf(&circlebuf,n);if(circlebuf.write>0) writeptr=circlebuf.write-1;else writeptr= Maxbuf-1;sprintf(writehistory[writehistorycount++],"生产者%d:缓冲区%d=%d", *n,writeptr,*n);sprintf(history[historycount++],"生产者%d:缓冲区%d=%d\n", *n,writeptr, *n);sem_post(&mutex);sem_post(&full);sleep(1);}
}
void * consumerThread(void *i)
{int *n=(int *)i;int t=TimesOfOp;						int value=0;							int readptr;							while(t--){sem_wait(&full);						sem_wait(&mutex);value=readCirclebuf(&circlebuf);			if(circlebuf.read>0) readptr=circlebuf.read-1;	else readptr= Maxbuf-1;sprintf(readhistory[readhistorycount++], "消费者%d:缓冲区%d=%d\n", *n,readptr,value);sprintf(history[historycount++],"消费者%d:缓冲区%d=%d\n", *n,readptr,value);sem_post(&mutex);						sem_post(&empty);						sleep(1);							}
}
int main()
{int i,max;int ConsNum=0,ProdNum=0,ret;sem_init(&mutex,0,1);sem_init(&empty,0,Maxbuf);sem_init(&full,0,0);	signal(SIGINT, sigend);signal(SIGTERM, sigend);circlebuf.read=circlebuf.write=0;for(i=0;i<Maxbuf;i++)circlebuf.buf[i]=0;printf("请输入生产者线程的数目 :");scanf("%d",&ProdNum);int *pro=(int*)malloc(ProdNum*sizeof(int));pthread_t *proid=(pthread_t*)malloc(ProdNum*sizeof(pthread_t));printf("请输入消费者线程的数目 :");scanf("%d",&ConsNum);int *con=(int*)malloc(ConsNum*sizeof(int));pthread_t *conid=(pthread_t*)malloc(ConsNum*sizeof(pthread_t));for(i=1;i<=ConsNum;i++){con[i-1]=i;ret=pthread_create(&conid[i],NULL,consumerThread,(void *)&con[i-1]);if(ret!=0){printf("Create thread error");exit(1);}}for(i=1;i<=ProdNum;i++){pro[i-1]=i;ret=pthread_create(&proid[i],NULL,productThread,(void *)&pro[i-1]);if(ret!=0){printf("Create thread error");exit(1);}}sleep((ConsNum+ ProdNum)*10);if (writehistorycount>readhistorycount) max=writehistorycount;else max=readhistorycount;for(i=0;i<max;i++)if ((i<writehistorycount) && (i<readhistorycount))printf("%s | %s\n",writehistory[i],readhistory[i]);else if (i<writehistorycount)printf("%s | %s\n",writehistory[i]," ");else 	printf("%s | %s\n"," ",readhistory[i]);printf("*************缓冲池的操作历史为:******************\n");for(i=0;i<historycount;i++)  printf("%s",history[i]);sem_destroy(&mutex);	sem_destroy(&empty);sem_destroy(&full);
}

编译链接命令:gcc pc1.c -o pc1 -lpthread

运行命令:./pc1

交互与结果:

3、生产者-消费者未加同步与互斥机制的运行试验

源程序:

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <pthread.h>
#include <unistd.h>
#include <signal.h>
#define Maxbuf 10       //缓冲单元数目
#define TimesOfOp 10    //生产者、消费者循环读写缓冲区的次数
#define true 1
#define false 0
#define historynum 100  //生产者、消费者读写历史记录数目
struct Circlebuf    //循环缓冲队列结构
{int read;            //读指针int write;       //写指针int buf[Maxbuf];    //缓冲区
} circlebuf;char writehistory[historynum][30];//写历史
char readhistory[historynum][30];//读历史
int writehistorycount=0;        //写历史计数器
int readhistorycount=0;         //读历史计数器
char history[historynum][30];    //缓冲区操作历史
int historycount=0;        //缓冲区操作历史计数器void writeCirclebuf(struct Circlebuf *circlebuf,int *value) //向缓冲区中写一个值
{circlebuf->buf[circlebuf->write]=(*value);sleep(1);                           circlebuf->write=(circlebuf->write+1)%Maxbuf; 
}
int readCirclebuf(struct Circlebuf *circlebuf)
{int value=0;value=circlebuf->buf[circlebuf->read];        sleep(1);                                circlebuf->buf[circlebuf->read]=0;            circlebuf->read=(circlebuf->read+1)%Maxbuf;     return value;
}void sigend(int sig)
{exit(0);
}void * productThread(void *i)
{int *n=(int *)i;int t=TimesOfOp;int writeptr;while(t--){   writeCirclebuf(&circlebuf,n);if(circlebuf.write>0) writeptr=circlebuf.write-1;else writeptr= Maxbuf-1;sprintf(writehistory[writehistorycount++],"生产者%d:缓冲区%d=%d", *n,writeptr,*n);sprintf(history[historycount++],"生产者%d:缓冲区%d=%d\n", *n,writeptr, *n);sleep(1);}
}void * consumerThread(void *i)
{int *n=(int *)i;int t=TimesOfOp;                        int value=0;                            int readptr;                            while(t--){value=readCirclebuf(&circlebuf);            if(circlebuf.read>0) readptr=circlebuf.read-1;    else readptr= Maxbuf-1;sprintf(readhistory[readhistorycount++], "消费者%d:缓冲区%d=%d\n", *n,readptr,value);sprintf(history[historycount++],"消费者%d:缓冲区%d=%d\n", *n,readptr,value);sleep(1);}
}int main()
{int i,max;int ConsNum=0,ProdNum=0,ret;circlebuf.read=circlebuf.write=0;for(i=0;i<Maxbuf;i++)circlebuf.buf[i]=0;printf("请输入生产者线程的数目 :");scanf("%d",&ProdNum);int *pro=(int*)malloc(ProdNum*sizeof(int));pthread_t *proid=(pthread_t*)malloc(ProdNum*sizeof(pthread_t));printf("请输入消费者线程的数目 :");scanf("%d",&ConsNum);int *con=(int*)malloc(ConsNum*sizeof(int));pthread_t *conid=(pthread_t*)malloc(ConsNum*sizeof(pthread_t));for(i=1;i<=ConsNum;i++){con[i-1]=i;ret=pthread_create(&conid[i],NULL,consumerThread,(void *)&con[i-1]);if(ret!=0){printf("Create thread error");exit(1);}}for(i=1;i<=ProdNum;i++){pro[i-1]=i;ret=pthread_create(&proid[i],NULL,productThread,(void *)&pro[i-1]);if(ret!=0){printf("Create thread error");exit(1);}}sleep((ConsNum+ ProdNum)*10);if (writehistorycount>readhistorycount) max=writehistorycount;else max=readhistorycount;for(i=0;i<max;i++)if ((i<writehistorycount) && (i<readhistorycount))printf("%s | %s\n",writehistory[i],readhistory[i]);else if (i<writehistorycount)printf("%s | %s\n",writehistory[i]," ");else    printf("%s | %s\n"," ",readhistory[i]);printf("*************缓冲池的操作历史为:******************\n");for(i=0;i<historycount;i++)  printf("%s",history[i]);
}

编译链接命令:gcc pc2.c -o pc2 -lpthread

运行命令:./pc2

交互与结果:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/21015.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

图解 Python 编程(10) | 错误与异常处理

&#x1f31e;欢迎来到Python的世界 &#x1f308;博客主页&#xff1a;卿云阁 &#x1f48c;欢迎关注&#x1f389;点赞&#x1f44d;收藏⭐️留言&#x1f4dd; &#x1f31f;本文由卿云阁原创&#xff01; &#x1f4c6;首发时间&#xff1a;&#x1f339;2024年6月2日&…

LangChain学习之prompt格式化与解析器使用

1. 学习背景 在LangChain for LLM应用程序开发中课程中&#xff0c;学习了LangChain框架扩展应用程序开发中语言模型的用例和功能的基本技能&#xff0c;遂做整理为后面的应用做准备。视频地址&#xff1a;基于LangChain的大语言模型应用开发构建和评估高 2. 先准备尝试调用O…

数据结构(C):从初识堆到堆排序的实现

目录 &#x1f31e;0.前言 &#x1f688; 1.堆的概念 &#x1f688; 2.堆的实现 &#x1f69d;2.1堆向下调整算法 &#x1f69d;2.2堆的创建&#xff08;堆向下调整算法&#xff09; ✈️2.2.1 向下调整建堆时间复杂度 &#x1f69d;2.3堆向上调整算法 &#x1f69d;2.…

testcontainer

在我们的项目中&#xff0c;单元测试是保证我们代码质量非常重要的一环&#xff0c;但是我们的业务代码不可避免的需要依赖外部的系统或服务如DB&#xff0c;redis&#xff0c;其他外部服务等。如何保证我们的测试代码不受外部依赖的影响&#xff0c;能够稳定的运行成为了一件比…

c++------类和对象(下)包含了this指针、构造函数、析构函数、拷贝构造等

文章目录 前言一、this指针1.1、this指针的引出1.2、 this指针的特性 二、类的默认的六个构造函数2.1、构造函数简述2.2构造函数 三、析构函数3.1、析构函数引出3.2、特点&#xff1a; 四、拷贝构造4.1、引入4.2、特征&#xff1a;4.3、默认拷贝构造函数 总结 前言 在本节中&a…

中国的历史看中国的经济发展

从中国的历史看中国的经济发展&#xff0c;可以发现其经历了几个显著的阶段&#xff0c;每个阶段都有其独特的特点和成就&#xff1a; 古代经济&#xff1a;中国古代经济以农业为主&#xff0c;实行井田制&#xff0c;重视水利工程的建设&#xff0c;如都江堰、灵渠等。 商业发…

Compose Multiplatform 1.6.10 发布,解释一些小问题, Jake 大佬的 Hack

虽然一直比较关注跨平台开发&#xff0c;但其实我很少写 Compose Multiplatform 的内容&#xff0c;因为关于 Compose Multiplatform 的使用&#xff0c;其实我并没在实际生产环境上发布过&#xff0c;但是这个版本确实值得一提&#xff0c;因为该版本包含&#xff1a; iOS Bet…

数据库(15)——DQL分页查询

DQL分页查询语法 SELECT 字段列表 FROM 表名 LIMIT 起始索引&#xff0c;查询记录数; 注&#xff1a;起始索引从0开始&#xff0c;起始索引&#xff08;查询页码-1&#xff09;*每页显示记录数。 如果查询的是第一页&#xff0c;可以省略起始索引。 示例&#xff1a;查询第一页…

【考研数学】概率论如何复习?跟谁好?

概率论一定要跟对老师&#xff0c;如果跟对老师&#xff0c;考研基本上能拿满分 概率论在考研试卷中占比并不大&#xff0c;其中&#xff1a; 高等数学&#xff0c;90分&#xff0c;约占比60%; 线性代数&#xff0c;30分&#xff0c;约占比20%; 概率论与数理统计&#xff0…

hive中的join操作及其数据倾斜

hive中的join操作及其数据倾斜 join操作是一个大数据领域一个常见的话题。归根结底是由于在数据量超大的情况下&#xff0c;join操作会使内存占用飙升。运算的复杂度也随之上升。在进行join操作时&#xff0c;也会更容易发生数据倾斜。这些都是需要考虑的问题。 过去了解到很…

每日5题Day15 - LeetCode 71 - 75

每一步向前都是向自己的梦想更近一步&#xff0c;坚持不懈&#xff0c;勇往直前&#xff01; 第一题&#xff1a;71. 简化路径 - 力扣&#xff08;LeetCode&#xff09; class Solution {public String simplifyPath(String path) {Deque<String> stack new LinkedList…

mysql的增删查改(进阶)

目录 一. 更复杂的新增 二. 查询 2.1 聚合查询 COUNT SUM AVG MAX MIN 2.1.2 分组查询 group by 子句 2.1.3 HAVING 2.2 联合查询/多表查询 2.2.1 内连接 2.2.2 外连接 2.2.3 全外连接 2.2.4 自连接 2.2.5 子查询 2.2.6 合并查询 一. 更复杂的新增 将从表名查询到…

自动化办公01 smtplib 邮件⾃动发送

目录 一、准备需要发送邮件的邮箱账号 二、发送邮箱的基本步骤 1. 登录邮箱 2. 准备数据 3. 发送邮件 三、特殊内容的发送 1. 发送附件 2. 发送图片 3. 发送超文本内容 4.邮件模板内容 SMTP&#xff08;Simple Mail Transfer Protocol&#xff09;即简单邮件传输协议…

霍夫曼树教程(个人总结版)

背景 霍夫曼树&#xff08;Huffman Tree&#xff09;是一种在1952年由戴维霍夫曼&#xff08;David A. Huffman&#xff09;提出的数据压缩算法。其主要目的是为了一种高效的数据编码方法&#xff0c;以便在最小化总编码长度的情况下对数据进行编码。霍夫曼树通过利用出现频率…

【Qt秘籍】[009]-自定义槽函数/信号

自定义槽函数 在Qt中自定义槽函数是一个直接的过程&#xff0c;槽函数本质上是类的一个成员函数&#xff0c;它可以响应信号。所谓的自定义槽函数&#xff0c;实际上操作过程和定义普通的成员函数相似。以下是如何在Qt中定义一个自定义槽函数的步骤&#xff1a; 步骤 1: 定义槽…

<jsp:setProperty>设置有参构造函数创建的自定义对象的属性

假设某一个类&#xff08;如TextConverter类&#xff09;有一个无参构造函数和一个有参构造函数&#xff0c;我们可以在Servlet里面先用有参构造函数自己new一个对象出来&#xff0c;存到request.setAttribute里面去。 Servlet转发到jsp页面后&#xff0c;再在jsp页面上用<j…

django基于大数据+Spring的新冠肺炎疫情实时监控系统设计和实现

设计一个基于Django(后端)和Spring(可能的中间件或服务集成)的新冠肺炎疫情实时监控系统涉及多个方面,包括数据收集、数据处理、数据存储、前端展示以及可能的中间件服务(如Spring Boot服务)。以下是一个大致的设计和实现步骤: 1. 系统架构 前端:使用Web框架(如Reac…

三种字符串的管理方式

NSString的三种实现方式 OC这个语言在不停的升级自己的内存管理&#xff0c;尽量的让自己的 OC的字符串 问题引入 在学习字符串的过程中间会遇到一个因为OC语言更新造成的问题 例如&#xff1a; int main(int argc, const char * argv[]) {autoreleasepool {NSString* str1 …

C++核心编程类的总结封装案例

C类的总结封装案例 文章目录 C类的总结封装案例1.立方体类的封装2.点与圆的关系的封装3.总结 1.立方体类的封装 在C中&#xff0c;我们可以定义一个立方体&#xff08;Cube&#xff09;类来封装立方体的属性和方法。立方体的属性可能包括边长&#xff08;side length&#xff…

【redis】set和zset常用命令

set 无序集合类型 sadd 和 smembers SADD&#xff1a;将一个或者多个元素添加到set中。注意,重复的元素无法添加到set中。 语法&#xff1a;SADD key member [member] 把集合中的元素,叫做member,就像hash类型中,叫做field类似. 返回值表示本次操作,添加成功了几个元素. 时间复…