Linux利用信号量实现线程的同步与互斥

线程使用互斥锁可以实现线程间的互斥,而互斥锁本身就是对资源的一种标识状态,当可以申请到锁时说明此时资源可以使用,当申请锁失败时说明资源此时被其他线程所占用不可使用,我们可以使用信号量来代替互斥锁实现。
信号量用来表示资源数目,当一个线程要去访问资源时,必须先去申请信号量,如果可以获取到信号量那么该线程就可以正常运行,如果获取资源失败那么就需要去等待。
当一个线程获取到信号量并且执行完成后必须去释放对应的信号量来通知其他线程进行工作,也就是说使用信号量不仅可以实现线程的互斥也可以实现线程间的同步。

信号量的操作:
①初始化
int sem_init(sem_t *sem, int pshared, unsigned int value);
参数描述:
sem:用来标识那个信号量,是一个sem_t类型的值
pshared:pshared为0,表示信号量作用于同一进程下的线程间同步
value:信号量的初始值
返回值:
成功返回0, 失败返回错误码
②销毁信号量
int sem_destroy(sem_t *sem);
参数描述:
sem:用来标识是哪个信号量
返回值;
成功返回0,失败返回错误码
③获得资源操作(P操作)
int sem_wait(sem_t *sem);
函数说明:
wait操作,使信号量的值减1,如果已经是0,则该线程被挂起到等待队列。
参数描述:
sem:用来标识啥事那个信号量
返回值:
成功返回0, 失败返回错误码
int sem_trywait(sem_t *sem);
该函数与上面的函数作用一样,不过这个函数不会将线程挂起等待。
④释放资源操作(V操作)
int sem_post(sem_t *sem);
函数说明:
psot操作,使信号量的值加1,同时唤醒等待该信号量的线程。
参数描述:
sem:标识是哪个信号量
返回值:
成功返回0, 失败返回错误码

实例:
将环形队列设置为缓冲区,生产者每生产一个数据放入到缓冲区中,消费者每次从缓冲区中读出一个数据。
此时需要使用两个信号量,一个用来表示环形队列中空位置的数量,一个用来表示环形队列中数据的数量。

生产者只有等到有空位置时才能生产数据,消费者只有等到有数据时才能取到数据

#include<stdio.h>
#include<semaphore.h>
#include<pthread.h>

//用数组模拟环形队列
int arr[8] = {0};
int i = 0;
int j = 0;
//设置信号量用来表示数组中的空位
sem_t blank;
//设置信号量用来表示数组中的数据
sem_t data;

void *product(void *arg)
{
    while(1)
    {
    //生产者要生产必须要有空位,不然只能等待消费者取走数据之后才能生产
    sem_wait(&blank);
    arr[i%8] = rand()%123;
    printf("生产者生产完成:%d\n", arr[i%8]);
    i++;
    //当生产成功时,数据量+1
    sem_post(&data);
    sleep(i%3);
    }
}

void *consumer(void *arg)
{
    while(1)
    {
    //消费者要消费必须有数据,不然只能等待生产者生产出数据之后才能取走
    sem_wait(&data);
    printf("消费者消费完成:%d\n", arr[j%8]);
    j++;
    //当消费成功时,空位量+1
    sem_post(&blank);
    sleep(j%5);
    }
}

int main()
{
    //最开始,有8个空位
    sem_init(&blank, 0, 8);
    //最开始,还没有放入数据,数据为0个
    sem_init(&data, 0, 0);
    pthread_t thread1;
    pthread_t thread2;

    pthread_create(&thread1, NULL, &product, NULL);
    pthread_create(&thread2, NULL, &consumer, NULL);

    pthread_join(thread1, NULL);
    pthread_join(thread2, NULL);

    sem_destroy(&blank);
    sem_destroy(&data);
    return 0;
}


  • 2
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
<h3>回答1:</h3><br/>Linux线程同步互斥是指在多个线程同时访问共享资源时,为了避免数据竞争和不一致性而采取的一些措施。同步是指多个线程按照一定的顺序执行,以保证数据的正确性和一致性;互斥是指在同一时只允许一个线程访问共享资源,以避免多个线程同时修改数据而导致的错误。在Linux系统中,常用的同步互斥机制包括信号量互斥锁、条件变量等。这些机制可以通过系统调用或者库函数来实现,从而保证多线程程序的正确性和稳定性。 <h3>回答2:</h3><br/>在多线程编程中,线程同步互斥是非常重要的一个话题。在Linux系统下,线程同步互斥通常使用pthread库提供的相关函数来实现。本文将重点介绍pthread库中几个常用的线程同步互斥的相关函数。 线程同步: 1. pthread_cond_init/pthread_cond_destroy 这两个函数主要用于条件变量的初始化和销毁。条件变量是线程同步的一种方式,用于在线程传递信息。 2. pthread_cond_signal/pthread_cond_broadcast 这两个函数用于唤醒一个或多个等待的线程。 3. pthread_cond_wait 这个函数用于阻塞当前线程,直到条件变量被唤醒。 线程互斥: 1. pthread_mutex_init/pthread_mutex_destroy 这两个函数主要用于互斥量的初始化和销毁。互斥量是一种保护共享资源的一种机制。 2. pthread_mutex_lock/pthread_mutex_trylock/pthread_mutex_unlock 这三个函数分别用于加锁、尝试加锁和解锁互斥量。 以上这些函数是pthread库中最常用的线程同步互斥相关函数。在实际编程中,经常使用这些函数来保证线程互斥同步。另外,还有很多其他的线程同步互斥的方式,如信号量、读写锁等等。需要根据具体的应用场景和需求来选择。 <h3>回答3:</h3><br/>线程同步互斥是多线程程序开发中重要的概念。在多个线程同时访问共享资源时,为了保证数据的正确性和一致性,需要使用线程同步互斥技术来防止数据竞争(Data Race)。 线程同步是指协调多个线程的执行顺序,让它们以正确的顺序访问共享资源。线程同步的方法有很多种,比如信号量互斥锁、条件变量等。其中,互斥锁(Mutex)是最常用的一种同步方法。 互斥锁是用来保护共享资源,实现线程互斥的一种机制。在临界区(Critical Section)内,只能有一个线程访问共享资源。当一个线程进入临界区时,首先要尝试获得互斥锁。如果锁已经被其他线程占用,则当前线程会被阻塞,直到其他线程释放锁。当当前线程执行完临界区的代码后,需要释放互斥锁,让其他线程获得资源的使用权。 互斥锁的实现方式有很多种,比如基于原子操作、自旋锁、互斥量等。在Linux系统中,最常用的是互斥量(pthread_mutex_t)。互斥锁可以确保临界区内的代码在同一时刻只会被一个线程执行,从而避免了数据竞争所带来的副作用。 除了互斥锁,Linux系统还提供了其他同步机制,比如条件变量(pthread_cond_t)和信号量(sem_t)。这些机制的主要作用是在多个线程传递信号和状态,从而实现线程同步。 总之,在多线程编程中,线程同步互斥是必不可少的。只有通过合适的同步措施,才能保证多个线程能够正确地协作,从而充分利用多核CPU的计算能力,提高程序的性能和响应速度。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值