Linux中线程的同步与互斥、生产者消费模型和读者写者问题、死锁问题

线程的同步与互斥

线程是一个存在进程中的一个执行控制流,因为线程没有进程的独立性,在进程内部线程的大部分资源数据都是共享的,所以在使用的过程中就需要考虑到线程的安全和数据的可靠。不能因为线程之间资源的竞争而导致数据发生错乱,也不能因为有些线程因为调度器长时间没有调度从而导致饥饿问题。所以在线程中也有了同步与互斥,这里用 “也” 是因为进程中也有同步与互斥,今天来了解线程中的同步与互斥。

互斥量

我们为什么要有互斥量

首先,一个进程中的多个线程因为同处于一个虚拟地址空间中,所以相互之间大部分数据是共享的,从而在线程竞争中出现数据错乱的情况,我们来举例子来看一下

首先我们采用多个线程去竞争着去修改count的数据,我们定义初始的count为100,让其减少为0就可以了,我们看看会有什么结果。

如果对线程不了解或者相关的创建API点击线程概念、线程创建、等待、退出

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

#define num 3 //我们用3个线程

int count = 100;

void* MyFun(void* arg) //线程入口函数
{
    int j = (int)arg;
    while (ticket >= 0)
    {
        usleep(1000);
        printf("thread %d count %d\n", j, count);
        count--;
    }
    return NULL;
}

int main()
{
    pthread_t th[num];
    int i = 0;
    for (;i < num; ++i)
    {
        // 创建线程
        int ret = pthread_create(&th[i], NULL, MyFun, (void*)i);
        if (ret != 0)
        {
            perror("pthread_create error\n");
            exit(1);
        }
    }

    i = 0;
    for (; i < num; ++i)
    {
        // 等待线程,释放资源
        pthread_join(th[i], NULL);
    }
    return 0;
}

我们先看看结果
线程不安全
看到这个结果,如果仔细看代码就会发现,我们写的是代码count到0就退出,但是为什么会出现到-2呢?这里我们就解释一下

首先线程之间资源共享,所以随着cpu的调度,如果你是多核计算机,线程是可以同时访问一个共享资源,这个时候就会发生:
内存数据到寄存器到cpu
这只是其中的一种情况,还有可能在内存读到寄存器中的时候被切换掉等,就出现了数据错误的情况。
所以我们要加上互斥锁,让互斥的访问临界资源。
我们修改我们的代码:

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

#define num 3

// 互斥锁
pthread_mutex_t mutxe;

int ticket = 100;

void* MyFun(void* arg)
{
    long j = (long)arg;
    while (1)
    {
        usleep(1000);
        pthread_mutex_lock(&mutxe); // 加锁
        if (ticket > 0)
        {
            printf("thread %lu ticket %d\n", j, ticket);
            ticket--;
            pthread_mutex_unlock(&mutxe); // 解锁
    
  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
生产者消费者模型是一个经典的并发编程问题,在多线程环境下需要进行进程同步互斥。其生产者负责向缓冲区放置数据,而消费者则负责从缓冲区取出数据。 在实验,我们可以通过使用信号量来实现进程同步互斥。具体地说,我们可以使用两个信号量:一个用于表示缓冲区是否为空,另一个用于表示缓冲区是否已满。 当生产者向缓冲区放入数据时,它会获取“缓冲区是否已满”的信号量。如果缓冲区已满,则生产者需要等待,直到有空可用。一旦缓冲区有空生产者就会向缓冲区放入数据,并释放“缓冲区是否为空”的信号量。 当消费者从缓冲区取出数据时,它会获取“缓冲区是否为空”的信号量。如果缓冲区为空,则消费者需要等待,直到有数据可用。一旦缓冲区有数据,消费者就会从缓冲区取出数据,并释放“缓冲区是否已满”的信号量。 通过这种方式,我们可以确保生产者消费者能够正确地交替执行,并且不会出现数据竞争和死锁问题。 下面是一个生产者消费者模型的实验结果截图: ![生产者消费者模型实验结果截图](https://img-blog.csdnimg.cn/20211210145604663.png) 从截图可以看出,在多线程环境下,生产者消费者能够正确地交替执行,并且不会出现数据竞争和死锁问题。同时,缓冲区的大小也可以根据需要进行调整,以满足不同的需求。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值