首页 > 其他分享 >高编:线程(2)——同步与互斥

高编:线程(2)——同步与互斥

时间:2024-07-01 20:59:01浏览次数:15  
标签:int 信号量 互斥 mutex pthread 线程 sem 高编

一、互斥

概念:
    互斥 ===》在多线程中对 临界资源 的 排他性访问。

    互斥机制 ===》互斥锁  ===》保证临界资源的访问控制。

    pthread_mutex_t   mutex;
    互斥锁类型            互斥锁变量 内核对象

框架:
     定义互斥锁 ==》初始化锁 ==》加锁 ==》解锁 ==》销毁
        ****                                           ***           ***

     1、定义

      pthread_mutex_t   mutex;

     2、初始化锁

        int pthread_mutex_init(pthread_mutex_t *mutex,const pthread_mutexattr_t *attr);
        功能:将已经定义好的互斥锁初始化。
        参数:mutex  要初始化的互斥锁
                   atrr  初始化的值,一般是NULL表示默认锁
        返回值:成功 0
                      失败 非零

    3、加锁

        int pthread_mutex_lock(pthread_mutex_t *mutex);
        功能:用指定的互斥锁开始加锁代码
                   加锁后的代码 到 解锁部分的代码 属于 原子操作(计算机中不可再分割的操作),
                   在加锁期间 其他进程/线程 都不能 操作该部分代码
                   如果该函数在执行的时候,mutex已经被其他部分使用 则 代码阻塞。

        参数: mutex 用来给代码加锁的互斥锁
        返回值:成功  0
                      失败  非零

     4、解锁

        int pthread_mutex_unlock(pthread_mutex_t *mutex);
        功能:将指定的互斥锁解锁。
                   解锁之后代码不再排他访问,一般加锁解锁同时出现。(尽量上锁解锁的代码短一些,保护并发性)
        参数:mutex  用来解锁的互斥锁
        返回值:成功  0
                      失败  非零

     5、销毁         

        int pthread_mutex_destroy(pthread_mutex_t *mutex);
         功能:使用互斥锁完毕后需要销毁互斥锁
         参数:mutex  要销毁的互斥锁
         返回值:成功  0
                       失败  非零

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <pthread.h>

int a = 0;
pthread_mutex_t mutex;
void* th(void* arg)
{
    // pthread_mutex_lock(&mutex);
    int i =5000;
    while(i--)
    {
        pthread_mutex_lock(&mutex);
        int tmp = a;
        printf("a is %d\n",tmp+1);
        a = tmp+1;
        pthread_mutex_unlock(&mutex); 
    }
    // pthread_mutex_unlock(&mutex); 
    return NULL;
}

int main(int argc, char *argv[])
{
    
    pthread_t tid1,tid2;
    pthread_mutex_init(&mutex,NULL);
    pthread_create(&tid1,NULL,th,NULL);
    pthread_create(&tid2,NULL,th,NULL);
    pthread_join(tid1,NULL);
    pthread_join(tid2,NULL);
    pthread_mutex_destroy(&mutex);
    return 0;
}

 

6、trylock

        int pthread_mutex_trylock(pthread_mutex_t *mutex);
        功能:类似加锁函数效果,唯一区别就是不阻塞。
        参数:mutex 用来加锁的互斥锁
        返回值:成功  0
                      失败  非零
                       E_AGAIN

 


综合练习:10个人去三个窗口办业务,办完一个人才能开始办下一个人

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <pthread.h>
pthread_mutex_t mutex;
int WIN = 3;
void* th(void* arg)
{
    while(1)
    {
        pthread_mutex_lock(&mutex);
        if(WIN>0)
        {
            WIN--;
            pthread_mutex_unlock(&mutex);
            printf("get win\n");
            sleep(rand()%5);

            printf("relese win\n");
            pthread_mutex_lock(&mutex);
            WIN++;
            pthread_mutex_unlock(&mutex);

            break;
        }
        else 
        {
            pthread_mutex_unlock(&mutex);

        }
    }
    return NULL;
}

int main(int argc, char *argv[])
{
    pthread_t tid[10];
    int i = 0 ;
    pthread_mutex_init(&mutex,NULL);
    for(i = 0 ;i<10;i++)
    {
        pthread_create(&tid[i],NULL,th,NULL);
    }
    for(i = 0 ;i<10;i++)
    {
        pthread_join(tid[i],NULL);
    }
    pthread_mutex_destroy(&mutex);
    return 0;
}

 二、同步

        同步 ===》有一定先后顺序的对资源的排他性访问。

    原因:互斥锁可以控制排他访问但没有次序。

    linux下的线程同步  ===》信号量机制 ===》semaphore.h   posix 
    sem_open();
    信号量的分类:
    1、无名信号量 ==》线程间通信
    2、有名信号量 ==》进程间通信

    框架:
    信号量的定义 ===》信号量的初始化 ===》信号量的P(申请资源)V(释放资源)操作 

===》信号量的销毁。

    semaphore 

1、信号量的定义 

        sem_t                sem;
       信号量的类型     信号量的变量

2、信号量的初始化

        int sem_init(sem_t *sem, int pshared, unsigned int value);
        功能:将已经定义好的信号量赋值。
        参数:sem 要初始化的信号量
                   pshared = 0 ;表示线程间使用信号量
                                 !=0 ;表示进程间使用信号量
                   value 信号量的初始值,一般无名信号量
                            都是二值信号量,0 1 
                            0 表示红灯,进程暂停阻塞
                            1 表示绿灯,进程可以通过执行
        返回值:成功  0
                      失败  -1;

3、信号量的PV 操作

       P ===》申请资源===》申请一个二值信号量 
       V ===》释放资源===》释放一个二值信号量

       P操作对应函数 ==》sem_wait();
       V操作对应函数 ==》sem_post();


    int sem_wait(sem_t *sem);//会阻塞
    功能:判断当前sem信号量是否有资源可用。
              如果sem有资源(==1),则申请该资源,程序继续运行
              如果sem没有资源(==0),则线程阻塞等待,一旦有资源,则自动申请资源并继续运行程序。

     注意:sem 申请资源后会自动执行 sem = sem - 1;
     参数:sem 要判断的信号量资源
     返回值:成功 0 
                   失败 -1


     int sem_post(sem_t *sem);
    功能:函数可以将指定的sem信号量资源释放
              并默认执行,sem = sem + 1;
              线程在该函数上不会阻塞。
    参数:sem 要释放资源的信号量
    返回值:成功 0
                  失败 -1;

4、信号量的销毁

        int sem_destroy(sem_t *sem);
       功能:使用完毕将指定的信号量销毁
       参数:sem要销毁的信号量
       返回值:成功 0
                     失败  -1;

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <pthread.h>
#include <semaphore.h>
sem_t sem_C,sem_M;
void* th1(void* arg)
{
    int i = 5;
    while(i--)
    {
        sem_wait(&sem_C);
        printf("capper ");
        fflush(stdout);
        sem_post(&sem_M);
    }
    return NULL;
}
void* th2(void* arg)
{
    int i = 5;
    while(i--)
    {
        sem_wait(&sem_M);
        printf("made in future\n");
        sleep(1);
        sem_post(&sem_C);
    }
    return NULL;
}

int main(int argc, char *argv[])
{
    pthread_t tid1,tid2;

    sem_init(&sem_C,0,1);
    sem_init(&sem_M,0,0);

    pthread_create(&tid1,NULL,th1,NULL);
    pthread_create(&tid2,NULL,th2,NULL);

    pthread_join(tid1,NULL);
    pthread_join(tid2,NULL);

    sem_destroy(&sem_C);
    sem_destroy(&sem_M);
    return 0;
}


三、互斥锁和信号量的异同

同一时刻只有一个在走

1. 运行顺序--->互斥锁没有顺序;信号量有顺序
2. 锁的个数--->互斥锁只有1个锁;信号量一般大于2个锁(几个同步变量就几个锁)
3. 加锁解锁操作--->互斥锁同一个线程:同一个锁上锁解锁;信号量同一个线程不同锁:锁自己,释放下一个锁

四、死锁

1.产生死锁的原因

(1) 因为系统资源不足。
(2) 进程运行推进的顺序不合适。
(3) 资源分配不当等。
如果系统资源充足,进程的资源请求都能够得到满足,死锁出现的可能性就很低,否则
就会因争夺有限的资源而陷入死锁。其次,进程运行推进顺序与速度不同,也可能产生死锁。

2.产生死锁的四个必要条件(同时满足 产生死锁)

(1) 互斥条件:一个资源每次只能被一个进程使用。
(2) 请求与保持条件:一个进程因请求资源而阻塞时,对已获得的资源保持不放。
(3) 不剥夺条件:进程已获得的资源,在末使用完之前,不能强行剥夺。
(4) 循环等待条件:若干进程之间形成一种 头尾相接 的循环等待资源关系。
     

标签:int,信号量,互斥,mutex,pthread,线程,sem,高编
From: https://blog.csdn.net/weixin_71751116/article/details/140094391

相关文章

  • Java进阶学习|Day4.Java多线程,线程池
    文章目录了解多线程CPU进程(Process)线程多线程开发多线程优点实现方式继承Thread类实现Runnable接口实现Callable接口线程状态转换线程状态线程调度调整线程优先级线程睡眠线程等待线程让步线程加入线程唤醒线程同步线程同步方式多线程间通信线程池了解线程池定义......
  • 线程池原理
    3线程池原理3.1ctlprivatefinalAtomicIntegerctl=newAtomicInteger(ctlOf(RUNNING,0));是原子变量,同时记录线程池状态和线程池线程个数以32位为例,高三位表示状态,后面表示个数3.2状态Running:接受新任务并处理阻塞队列里任务shutdown:拒绝新任务但是处理队列任务......
  • spring 定时任务 使用线程池
    spring线程池定义packagecom.minex.icp.tool.conf;importlombok.extern.slf4j.Slf4j;importorg.springframework.context.annotation.Bean;importorg.springframework.context.annotation.Configuration;importorg.springframework.scheduling.annotation.AsyncConfi......
  • 独立线程按照推入的任务时间顺序执行任务
    一、模型背景有时,代码需要交个其他线程处理,也不关心执行的进度,但是关心执行的顺序。二、代码实现publicclassThreadTaskChanel:IDisposable{//创建任务队列privateConcurrentQueue<Action>_taskQueue=newConcurrentQueue<Action>();......
  • 高编:进程(2)
    一、父子进程的关系:   子进程是父进程的副本。子进程获得父进程数据段,堆,栈,正文段共享。      在fork之后,一般情况那个会先运行,是不确定的。如果非要确定那个要先运行,需要IPC机制。      区别:   1)fork的返回值   2)pid不同二、进程的终止:8......
  • Java中线程的run()和start()有什么区别?
    在Java中,run() 方法和 start() 方法在线程的使用中有重要区别:run() 方法:当您直接调用线程对象的 run() 方法时,它会在当前线程中执行 run() 方法中的代码,不会启动新的线程。这就相当于普通的方法调用,不会实现多线程的并发执行效果。start() 方法:调用线程对象的 ......
  • LINUX系统编程:多线程互斥
    目录1.铺垫2.线程锁接口的认识静态锁分配动态锁的分配互斥量的销毁互斥量加锁和解锁3.加锁版抢票4.互斥的底层实现1.铺垫先提一个小场景,有1000张票,现在有4个进程,这四个进程疯狂的去抢这1000张票,看看会发生什么呢?#include<iostream>#include<thread>#include......
  • 多线程详解
    多线程详解1.线程简介任务,进程,线程,多线程多任务吃饭的时候玩手机,,,现实之中大多这样同时做多件事情的例子,看起来是多个任务都在做,其实本质上我们的大脑只做了一件事多线程原来是一条路,慢慢因为车多了,道路堵塞,效率极低,为了提高使用的效率,能够充分利用道路,于是加了多个车道。......
  • 线程进程以及多线程多进程(超详解)
    目录前言一、什么是进程和线程进程(Process)线程(Thread)多线程(Multithreading)多进程(Multiprocessing)相互组合关系二、资源分配进程私有资源共享资源线程私有资源共享资源多进程私有资源共享资源多线程私有资源共享资源进程的共享和私有资源线......
  • 【C语言】线程同步
    【C语言】线程同步线程同步1.互斥锁2.读写锁3.条件变量4.信号量最后线程同步  线程同步是指在多线程的情况下,如果多个线程去访问共享资源,需要按照一定规则顺序依次去访问,保证共享资源的数据一致性。1.互斥锁互斥相关函数//互斥量pthread_mutex_tmutex;......