线程初探——生产者、消费者模式

线程初探——生产者、消费者模式

helllo,大家好,这里是追風者频道。今天我们来聊一聊生产者、消费者模式。后期需要采用该模式来进行协议栈抓包架构的搭建,通过多线程对数据包进行分任务处理。

生产者消费者模式结构以及问题

生产者消费者模式的结构:

  1. 生产者线程。
  2. 消费者线程。
  3. 缓存区(暂存区)。

生产者消费者模式的痛点:

  1. 线程同步,对缓存区的操作要进行同步(生产者无限生产,撑爆缓存区或抹除原始数据;消费者无限消费,获取到错误数据,导致程序出错)。
  2. 循环使用缓存区(当生产者生产的数据超出数组索引上限时,需要从0开始使用数组)。
  3. 可以线程安全访问的信号量计数器。

下图为生产者消费者模式模拟图:

生产者消费者模式

解决问题

我们可以通过计数器信号量来解决 线程同步问题,控制线程执行。通过记录索引,来控制缓存区的使用。

	#include <stdio.h>
#include "sys_plat.h"

static sys_sem_t sem;
static sys_mutex_t mutex;
static int count;

static char buffer[100];
static int write_index, read_index, total;
static sys_sem_t read_sem, write_sem;



void thread1_entry (void * arg) {
    for (int i = 0; i < 2 * sizeof(buffer); i++) {
        // 当信号量为0时,表示缓存区中无资源读取了,无限期等待。
        sys_sem_wait(read_sem, 0);
        char data = buffer[read_index++];

        if (read_index >= sizeof(buffer)) {
            read_index = 0;
        }

        sys_mutex_lock(total);
        total--;
        sys_mutex_unlock(total);

        sys_sem_notify(write_sem);
        
        plat_printf("thread1 read data=%d\t index=%d\n", data, read_index);
        sys_sleep(2000);
    }
    
    
    plat_printf("thread1 count is %d\n", count);
    while (1) {
        plat_printf("this is thread1: %s\n", (char *)arg);
        sys_sleep(1000);
        sys_sem_notify(sem);
    }
}

void thread2_entry (void * arg) {
    sys_sleep(1000);
    for (int i = 0; i < 2 * sizeof(buffer); i++) {
        // 当信号量为0时,表示缓存区中无容量了,无限期等待。
        sys_sem_wait(write_sem, 0);
        buffer[write_index++] = i;

        if (write_index >= sizeof(buffer)) {
            write_index = 0;
        }

        sys_mutex_lock(total);
        total++;
        sys_mutex_unlock(total);
        
        plat_printf("thread2 write data=%d\t index=%d\n", i, write_index);

        sys_sem_notify(read_sem);
    }

    plat_printf("thread2 count is %d\n", count);
    while (1) {
        sys_sem_wait(sem, 0);
        plat_printf("this is thread2: %s\n", (char *)arg);
    }
}

int main (void) {
    // 读取信号量计数器,从 0 开始。
    read_sem = sys_sem_create(0);
    // 写入信号量计数器,从 满 开始。
    write_sem = sys_sem_create(sizeof(buffer));

    sem = sys_sem_create(0);
    mutex = sys_mutex_create();

    sys_thread_create(thread1_entry, "A");
    sys_thread_create(thread2_entry, "B");
    
    printf("hello,network!");
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值