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

信号量

信号量可以用于进程之间的同步与互斥也可用于线程的同步与互斥,其本质就是操作系统中的P/V原语,通过P操作让进程挂起,V操作让挂起的进程继续运行

要用到的头文件:#include<semaphore.h>
(以下函数成功返回0,失败返回-1。)

初始化

int sem_init __P ((sem_t *_sem, int pshared, unsigned int value));

sem为指向信号量结构的一个指针;pshared不为0时此信号量在进程间共享,否则只能为当前进程的所有线程共享;value给出了信号量的初始值。

信号量P操作

增加信号量的值。当有线程阻塞在这个信号量上时,调用这个函数会使其中的一个线程不在阻塞
用来阻塞当前线程直到信号量sem的值大于0,解除阻塞后将sem的值减一,表明公共资源经使用后减少

int sem_wait(sem_t * sem);int sem_post(sem_t * sem);

信号量V操作

int sem_post(sem_t *sem)

信号量销毁

不使用信号量时,销毁信号量以释放其所占用资源

int sem_destroy(sem_t *sem)

线程互斥

创建两个线程,线程1对全局变量(初始值1)加1并显示出来,线程2对全局变量的值加倍并打印出来。通过信号量,可以让两个线程不会同时访问操作全局变量,实现线程的互斥,但无法控制线程的执行顺序

#include<pthread.h>
#include<stdio.h>
#include<semaphore.h>
#include<stdlib.h>
#include<errno.h>
#define MAX 100
static sem_t sem;
static int global=1;
void *t1_exe(void*arg){
    while (global<MAX)
    {
        sem_wait(&sem);
        printf("In thread 1 before increment global=%d\n",global);
        global++;
        printf("In thread 1 after increment global=%d\n",global);
        sem_post(&sem);
        sleep(5);
    }   
}
void *t2_exe(void*arg){
    while (global<MAX)
    {
        sem_wait(&sem);
        printf("In thread 2 before increment global=%d\n",global);
        global*=2;
        printf("In thread 2 after increment global=%d\n",global);
        sem_post(&sem);
        sleep(6);
    }  
}
void main(){
    pthread_t pid1,pid2;
    int error1,error2;
    if(sem_init(&sem,0,1)==-1){
        perror("sem initialized failed");
        exit(0);
    }
    error1=pthread_create(&pid1,NULL,t1_exe,NULL);
    error2=pthread_create(&pid2,NULL,t2_exe,NULL);
    if(error1!=0||error2!=0){
        printf("pthread_create failed");
        return;
    }
    pthread_join(pid1,NULL);
    pthread_join(pid2,NULL);
    sem_destroy(&sem);
    return;

}

在这里插入图片描述

线程同步

利用信号量,将上例线程执行顺序设置为线程1,线程2交替执行,实现线程的同步

#include<pthread.h>
#include<stdio.h>
#include<semaphore.h>
#include<stdlib.h>
#include<errno.h>
#define MAX 100
static sem_t sem1,sem2;
static int global=1;
void *t1_exe(void*arg){
    while (global<MAX)
    {
        sem_wait(&sem1);
        printf("In thread 1 before increment global=%d\n",global);
        global++;
        printf("In thread 1 after increment global=%d\n",global);
        sem_post(&sem2);
        sleep(5);
    }   
}
void *t2_exe(void*arg){
    while (global<MAX)
    {d
        sem_wait(&sem2);
        printf("In thread 2 before increment global=%d\n",global);
        global*=2;
        printf("In thread 2 after increment global=%d\n",global);
        sem_post(&sem1);
        sleep(6);
    }  
}
void main(){
    pthread_t pid1,pid2;
    int error1,error2;
    if(sem_init(&sem1,0,1)||sem_init(&sem2,0,0)==-1){
        perror("sem initialized failed");
        exit(0);
    }
    error1=pthread_create(&pid1,NULL,t1_exe,NULL);
    error2=pthread_create(&pid2,NULL,t2_exe,NULL);
    if(error1!=0||error2!=0){
        printf("pthread_create failed");
        return;
    }
    pthread_join(pid1,NULL);
    pthread_join(pid2,NULL);
    sem_destroy(&sem1);
    sem_destroy(&sem2);
    return;

}

在这里插入图片描述

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
<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的计算能力,提高程序的性能和响应速度。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值