Linux多线程C++版(九) 线程同步和互斥-----线程信号量

16 篇文章 2 订阅
9 篇文章 3 订阅

1.基本概念
  • 信号量从本质上是一个非负整数计数器,是共享资源的的数目,通常被用来控制对共享资源的访问。
  • 信号量可以实现线程的同步和互斥
  • 通过sem_post()和sem_wait函数对信号量进行加减操作从而解决线程的同步和互斥
  • 信号量数据类型 sem_t
2.信号量创建和销毁
//信号量的定义
sem_t sem;
int sem_t_init(sem_t *sem,int pshared,unsigned value);
int sem_t_destroy(sem_t *sem);
返回:成功返回0 出错返回错误编号
  • 参数
    • sem:信号量指针
    • pshared:是否可以跨进程使用,0为不共享,1为共享
    • value:信号量的初始值
3.信号量加和减操作
int sem_post(sem_t *sem); //功能:增加信号量的值
int sem_wait(sem_t *sem); //功能:减少信号量的值
int sem_trywait(sem_t *sem);//sem_wait()非阻塞版本
返回:成功返回0 出错返回错误编号
  • 调用sem_post() 一次信号量做 加1操作
  • 调用sem_wait() 一次信号量做 减1操作
  • 当线程调用sem_wait()后,若信号量的值小于0则线程阻塞。只有其他线程在调用sem_post()对信号量做加操作后并且其值大于或等于0时,阻塞的线程才能继续运行
4.代码理解信号量
/*
	创建三个子线程,使用信号量去控制三个子线程的运行顺序。
*/
#include<semaphore.h>
#include<pthread.h>
#include<stdio.h>
#include<stdlib.h>
//定义两个线程信号量
sem_t sem1,sem2;
void* a_fn(void *arg){
    sem_wait(sem1);//此时sem1为0,再减一 线程堵塞
    printf("thread a running\n");
    return(void*)0;
}
void* b_fn(void *arg){
    sem_wait(sem2);//此时sem2为0,再减一 线程堵塞
    printf("thread b running\n");
    //释放线程a,让阻塞的线程a继续操作
    sem_post(sem1)
    return(void*)0;
}
void* c_fn(void *arg){
    printf("thread c running\n");
    //释放线程b(此时sem2为0,加1)
    //让阻塞的线程b继续运行
    sem_post(sem2);
    return(void*)0;
}
int main(void){
    int err;
	pthread_t a,b,c;
    
    //线程信号量的初始化,初始值为0
    sem_t_init(&sem1,0,0);
    sem_t_init(&sem2,0,0);
    
    //创建线程
	if ((err = pthread_create(&a, null, a_fn, (void*)0)) != 0) {
		perror("pthread_create error");
	}
    if ((err = pthread_create(&b, null, b_fn, (void*)0)) != 0) {
		perror("pthread_create error");
	}
    if ((err = pthread_create(&c, null, c_fn, (void*)0)) != 0) {
		perror("pthread_create error");
	}
    
    //主线程要等三个子线程执行完毕
    pthread_join(a,null);
    pthread_join(b,null);
    pthread_join(c,null);
    //线程信号的销毁
   	sem_t_destroy(&sem1);
    sem_t_destroy(&sem2);
    return 0;
}
5.信号量实例银行账户取款----实现互斥

头文件 account.h

#ifndef __ACCOUNT_H__
#define __ACCOUNT_H__
#include <pthread.h>
#include <semaphore.h>
//银行账户结构体
typedef struct{
    int code;//卡号
    double balance;//余额
    sem_t sem;//线程信号量
    
}Account;
//创建账户
extern Account* creat_account(int code,double balance);
//销毁账户
extern void destory_account(Account *a);
//取款
extern double withdraw(Account *a,double amt);
//存款
extern double deposit(Account *a,double amt);
//查看账户余额
extern double get_balance(Account *a);
#endif

功能文件account.c

#include "account.h"
#include <malloc.h>
#include <assert.h>
#include <string.h>
#include <semaphore.h>
//创建账户
Account* creat_account(int code,double balance){
    //创建账户在堆当中
    Account *a = (Account*)malloc(sizeof(Account));
    assert(a != NULL);//用来判断a指针所需要开辟的空间,开辟成功了吗,成功就继续,否则就程序终止运行。
    a->code = code;
    a->balance = balance;
   //线程信号量的初始化,初始值为1
    sem_t_init(&a->sem,0,1);
    return a;
}
//销毁账户
void destory_account(Account *a){
    assert(a != NULL);
    //销毁线程信号量
    	sem_t_destroy(&a->sem);
    free(a);//释放空间
}
//取款
double withdraw(Account *a,double amt){
    assert(a != NULL);//判断指针a是否为空
    //使信号量减一  锁定
    sem_wait(&a->sem);
    if(amt < 0 || amt > a->balance){
        //使信号量加一  解锁
        sem_post(&a->sem);
        return 0.0;
    }
    double balance = a->balance;
    sleep(1);
    balance -= amt;
    a->balance = balance;
    //使信号量加一   解锁
    sem_post(&a->sem);
    return amt;
}
//存款
double deposit(Account *a,double amt){
   assert(a != NULL);
   //使信号量减一  锁定
   sem_wait(&a->sem);
   if(amt < 0){
       //使信号量加一   解锁
       sem_post(&a->sem);
       return 0.0;
   }
    double balance = a->balance;
    sleep(1);
    balance += amt;
    a->balance = balance;
    //使信号量加一   解锁
    sem_post(&a->sem);
    return amt;
}
//查看账户余额
double get_balance(Account *a){
    assert(a != NULL);
   	//使信号量减一  锁定
   	sem_wait(&a->sem);
    double balance = a->balance;
    //使信号量加一   解锁
    sem_post(&a->sem);
    return balanec;
}

执行文件 account_test.c

#include "account.h"
#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

//操作者的结构体
typedef struct{
    char  name[20];//操作者姓名
    Account *account;//账户
    double amt;//金额
}OperArg
//定义取款操作的线程运行函数
void* withdraw_fn(void *arg){
    OperArg *oa = (OperArg*)arg;
    //deposit存款函数
    double amt = deposit(oa->account,oa->amt);
    printf("%8s(0x%lx) deposit %f from account %d/n",oa->name,pthread_self(),amt,oa->account->code);
    return (void*)0;
}
//定义存款操作的线程运行函数
void* deposit_fn(void *arg){
    OperArg *oa = (OperArg*)arg;
    //withdraw取款函数
    double amt = withdraw(oa->account,oa->amt);
    printf("%8s(0x%lx) withdraw %f from account %d/n",oa->name,pthread_self(),amt,oa->account->code);
    return (void*)0;
}
//定义查看银行账户的线程运行函数
void* check_fn(void *arg){
    
}
int main(void){
    int err;
    pthread_t boy,girl;
    Account *a = create_account(10001,10000)//卡号,余额
        
    OperArg o1 , o2;//两个操作者
    strcpy(o1.name,"boy");//strcpy()把第二个参数值,放到第一个参数中
    o1.account = a;
    o1.amt = 10000;
    
    strcpy(o2.name,"girl");
    o2.account = a;
    o2.amt = 10000;
    
    //启动两个子线程(boy和girl)同时去操作同一个银行账户
    if((err = pthread_create(&boy,null,withdraw_fn,(void*)&o1)!=0){
        printf("pthread create error");
    }
    if((err = pthread_create(&girl,null,withdraw_fn,(void*)&o2)!=0){
        printf("pthread create error");
    }
    
    pthread_join(boy,null);
    pthread_join(girl,null);
       
    //查看账户余额
    printf("account balance %f/n",get_balance(a));
    //销毁账户
    destroy_account(a);
    return 0;
}

程序运行结果:
在这里插入图片描述

6.信号量实例计算和取结果----实现线程同步

思路:

在这里插入图片描述

#include<pthread.h>
#include<stdio.h>
#include<stdlib.h>
#include<semaphore.h>
/*
	一个线程赋值计算结果,一个线程负责获取结果
	当计算结果的线程没有执行完毕,获取结果的线程要等待(阻塞)
*/
typedef struct{
    int res;//存放运算结果
    sem_t sem//线程信号量
}Result;

//计算并将结果放置Result中的线程运行函数
void* set_fn(void *arg){
    int sum;
    for(int i =1;i<=100;i++){
        sum +=i;
    }
    //将结果存放到Result
    Result *r = (Result*)arg;
    r->res = sum;
    //对线程信号量 加一  
    sem_post(&r->sem);
    return(void*) 0;
}

//获得结果的线程运行函数
void* get_fn(void *arg){
    Result *r = (Result*)arg;
    
    //对线程信号量 减一  
    sem_wait(&r->sem);
    //去获取计算的结果
    int res = r->res;
    printf("ox%lx get sum is %d\n",pthread_self(),res);
    return(void*)0;
}

int main(void){
    int err;
	//定义线程标识符cal get
	pthread_t cal, get;
    
    //线程信号量初始化
    sem_init(&r.sem,0.0);
 
    //启动获取结果的线程
	if ((err = pthread_create(&get, NULL, get_fn, (void*)&r)) != 0) {
		perror("pthread_create error");
	}
    //启动计算结果的线程
    if ((err = pthread_create(&cal, NULL, set_fn, (void*)&r)) != 0) {
		perror("pthread_create error");
	}
    
    //主线程要等两个子线程执行完毕
    pthread_join(get,null);
    pthread_join(cal,null);
    
    //线程信号量的销毁
    sem_destroy(&r.sem);
    return 0;
}

程序运行结果:

在这里插入图片描述

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
这是一门linuxc++通讯架构实战课程,针对c/c++语言已经掌握的很熟并希望进一步深造以将来用c++linux下从事网络通讯领域/网络服务器的开发和架构工作。这门课程学习难度颇高但也有着极其优渥的薪水(最少30K月薪,最高可达60-80K月薪),这门课程,会先从nginx源码的分析和讲解开始,逐步开始书写属于自己的高性能服务器框架代码,完善个人代码库,这些,将会是您日后能取得高薪的重要筹码。本课程原计划带着大家逐行写代码,但因为代码实在过于复杂和精细,带着写代码可能会造成每节课至少要4~5小时的超长时间,所以老师会在课前先写好代码,主要的时间花费在逐行讲解这些代码上,这一点望同学们周知。如果你觉得非要老师领着写代码才行的话,老师会觉得你当前可能学习本门课程会比较吃力,请不要购买本课程,以免听不懂课程并给老师差评,差评也会非常影响老师课程的销售并造成其他同学的误解。 这门课程要求您具备下面的技能:(1)对c/c++语言掌握的非常熟练,语言本身已经不是继续学习的障碍,并不要求您一定熟悉网络或者linux;(2)对网络通讯架构领域有兴趣、勇于挑战这个高难度的开发领域并期望用大的付出换取高薪;在这门课程中,实现了一个完整的项目,其中包括通讯框架和业务逻辑框架,浓缩总结起来包括如下几点:(1)项目本身是一个极完整的多线程高并发的服务器程序;(2)按照包头包体格式正确的接收客户端发送过来的数据包, 完美解决收包时的数据粘包问题;(3)根据收到的包的不同来执行不同的业务处理逻辑;(4)把业务处理产生的结果数据包正确返回给客户端;本项目用到的主要开发技术和特色包括:(1)epoll高并发通讯技术,用到的触发模式是epoll中的水平触发模式【LT】;(2)自己写了一套线程池来处理业务逻辑,调用适当的业务逻辑处理函数处理业务并返回给客户端处理结果;(3)线程之间的同步技术包括互斥信号量等等;(4)连接池中连接的延迟回收技术,这是整个项目中的精华技术,极大程度上消除诸多导致服务器程序工作不稳定的因素;(5)专门处理数据发送的一整套数据发送逻辑以及对应的发送线程;(6)其他次要技术,包括信号、日志打印、fork()子进程、守护进程等等;

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Unknown To Known

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值