c语言中的多线程的实现

c语言中的多线程的实现,第1张


一、多线程的概念及程序实现 1.什么是多线程?

线程:(LWP)线程是轻量级的进程,进程是资源分配的最小单位,线程是调度的最小单位。


(同一个进程下)线程共用同一个进程的资源。


多线程在切换的时候要比多进程切换的时候效率高。


多进程的安全性要比多线程的安全性高,因为进程资源相互独立,线程的资源相关共享(当前进程)。


线程在创建的时候几乎不需要内存空间,只是占用了状态和指向正在执行内存的指针的资源。


在使用线程的时候需要安装man手册:sudo apt-get install manpages-posix manpages-posix-dev

2.多线程创建(第三方库函数)

int pthread_create(pthread_t *thread, const pthread_attr_t *attr,
              void *(*start_routine) (void *), void *arg);
功能:在进程中创建线程 -pthread 或 -lpthread

#include 
参数:
        @thread:线程号
        @attr:NULL,使用默认属性
        @start_routine:线程体
              void *task (void *arg)
              {
        
              }
        @arg:向线程体传参
返回值:成功返回0,非0就是失败


pthread_t pthread_self(void);
功能:在当前线程内获取线程号

#include 
返回值:成功返回线程号,不会失败

示例:

#include 
#include 
#include 
void *task(void *arg)
{
    //这是线程体
    printf("这是我创建的第一个线程tid = %ld\n",pthread_self());
}
int main(int argc, char const *argv[])
{
    pthread_t tid;
    //注意:这里的tid不能用于接受返回值,因为返回值不是线程号
    //成功返回0,失败返回非零
    //tid只能通过形参来接收
    if((pthread_create(&tid,NULL,task,NULL))!=0){
        fprintf(stderr,"pthread create error\n");
        exit(EXIT_FAILURE);
    }
    printf("这是主线程 = %ld\n",tid);
    while(1);
    return 0;
}
3.多线程执行过程及顺序

时间片轮询,上下文切换

注:线程间如果有全局变量的话,它们共用同一块内存空间,就没有写时拷贝的机制了。


示例:

#include 
#include 
#include 
#include 

int num=10;
void *task(void *arg)
{
    //这是线程体
    printf("这是我创建的第一个线程tid = %ld\n",pthread_self());
    while(1){
        printf("i am thread...\n");
        sleep(1);
    }
    //在线程中不能使用exit或_exit,因为它们会让进程结束,只要进程一结束所有的线程都退出
    //return不会结束一个进程,可以结束线程
    // exit(0);
    return NULL;
}
int main(int argc, char const *argv[])
{
    pthread_t tid;
    if((pthread_create(&tid,NULL,task,NULL))!=0){
        fprintf(stderr,"pthread create error\n");
        exit(EXIT_FAILURE);
    }
    printf("这是主线程 = %ld\n",tid);
    while(1){
        printf("i am process,num = %d\n",num++);
        sleep(1);
    }
    return 0;
}
4.线程的退出

void pthread_exit(void *retval);
功能:退出一个线程(进程是不会结束)

#include 
参数:
        @retval:退出的状态

示例:

#include 
#include 
#include 
#include 

int num=10;
void *task(void *arg)
{
    //这是线程体
    printf("这是我创建的第一个线程tid = %ld\n",pthread_self());
    while(1){
        printf("i am thread num = %d\n",num++);
        sleep(1);
        pthread_exit(NULL);
    }
 
}
int main(int argc, char const *argv[])
{
    pthread_t tid;
    if((pthread_create(&tid,NULL,task,NULL))!=0){
        fprintf(stderr,"pthread create error\n");
        exit(EXIT_FAILURE);
    }
    printf("这是主线程 = %ld\n",tid);
    while(1){
        printf("i am process num = %d\n",num++);
        sleep(1);
    }
    return 0;
}
5.给线程发信号

int pthread_cancel(pthread_t thread);
功能:取消一个线程

#include 
参数:
        @thread:tid
返回值:成功返回0,失败返回非0

示例:

#include 
#include 
#include 
#include 

 pthread_t tid1, tid2;
void *task1(void *arg)
{
    int num=5;
    while(--num){
        sleep(1);
    }
    pthread_cancel(tid2);
    while(1);
}
void *task2(void *arg)
{
    //不关心别人发的取消信号
    pthread_setcancelstate(PTHREAD_CANCEL_DISABLE,NULL);

    while(1){
        printf("i am task2\n");
        sleep(1);
    }

}
int main(int argc, char const *argv[])
{


    if ((pthread_create(&tid1, NULL, task1, NULL)) != 0)
    {
        fprintf(stderr, "pthread create error\n");
        exit(EXIT_FAILURE);
    }

    if ((pthread_create(&tid2, NULL, task2, NULL)) != 0)
    {
        fprintf(stderr, "pthread create error\n");
        exit(EXIT_FAILURE);
    }
    printf("这是主线程 tid1= %ld,tid2 = %ld\n", tid1,tid2);
    
    pthread_join(tid1,NULL);
    pthread_join(tid2,NULL);

    return 0;
}
6.线程资源回收

int pthread_join(pthread_t thread, void **retval);
功能:阻塞等待线程结束,回收它的资源

#include 
参数:
        @thread:线程号
        @retval:获取到pthread_exit()的参数
返回值:成功返回0,失败返回非0

示例:

#include 
#include 
#include 
#include 

int num=10;
//int tt=500;
void *task(void *arg)
{
    //这是线程体
    printf("这是我创建的第一个线程tid = %ld\n",pthread_self());
    while(1){
        printf("i am thread num = %d\n",num++);
        sleep(1);
        pthread_exit(NULL);
        //pthread_exit(&tt);
    }
 
}
int main(int argc, char const *argv[])
{
    pthread_t tid;
    //int *ww;
    if((pthread_create(&tid,NULL,task,NULL))!=0){
        fprintf(stderr,"pthread create error\n");
        exit(EXIT_FAILURE);
    }
    printf("这是主线程 = %ld\n",tid);

    pthread_join(tid,NULL);
    //pthread_join(tid,&ww);
    //printf("ww = %d\n",*ww);
    return 0;
}
7.标记线程为分离态

线程的默认状态是结合态,需要在主线程中调用pthread_join来回收资源,如果资源不回收当进程结束的时候会被进程的父进程回收。


但是可以将线程的状态标记为分离态,分离态的线程退出的时候资源会被自动回收。


int pthread_detach(pthread_t thread);
功能:标记线程为分离态,分离态的线程资源会被自动回收

#include 
参数:
        @thread:tid
返回值:成功返回0,失败返回非0

示例:

#include 
#include 
#include 
#include 

pthread_t tid1, tid2;
void *task1(void *arg)
{
    int num=10;
    while(--num){
        sleep(1);
    }
    pthread_cancel(tid2);
    pthread_exit(NULL);
}
void *task2(void *arg)
{
    while(1){
        printf("i am task2\n");
        sleep(1);
    }

}
int main(int argc, char const *argv[])
{


    if ((pthread_create(&tid1, NULL, task1, NULL)) != 0)
    {
        fprintf(stderr, "pthread create error\n");
        exit(EXIT_FAILURE);
    }

    if ((pthread_create(&tid2, NULL, task2, NULL)) != 0)
    {
        fprintf(stderr, "pthread create error\n");
        exit(EXIT_FAILURE);
    }
    printf("这是主线程 tid1= %ld,tid2 = %ld\n", tid1,tid2);
    
    pthread_detach(tid1);
    pthread_detach(tid2);
   // pthread_join(tid1,NULL);
    //pthread_join(tid2,NULL);
    while(1);
    return 0;
}

二、线程的同步互斥机制 1.线程互斥锁

在多线程中,所有的线程共用同一个全局变量,但是多个线程在 *** 作同一个变量的时候,就可能会出现冲突的现象,内核为了解决这一问题引入线程的互斥锁,获取到锁的线程可以 *** 作变量,获取不到锁资源的线程阻塞等待,没有执行的先后顺序,谁抢占锁成功谁执行。


应用场景实例:

#include 
#include 
int money=1000;

void *task1(void *arg)
{
    int get=50;
    while(1){
       // printf("我是张三我正在取钱50\n");
        sleep(1);
        money = money - get;

        if(money > 0){
            printf("张三取钱成功了,账户剩余金额 = %d\n",money);
        }else{
            printf("取钱失败,退出...\n");
            break;
        }

    }
    pthread_exit(NULL);
}
void *task2(void *arg)
{
    int get=100;
    while(1){
       // printf("我是李四我正在取钱100\n");
        sleep(1);
        money = money - get;

        if(money > 0){
            printf("李四取钱成功了,账户剩余金额 = %d\n",money);
            
        }else{
            printf("取钱失败,退出...\n");
            break;
        }

    }
    pthread_exit(NULL);
}
int main(int argc, char const *argv[])
{
    pthread_t tid1,tid2;
   
    if((pthread_create(&tid1,NULL,task1,NULL)) != 0){
        fprintf(stderr,"pthread create error\n");
        exit(EXIT_FAILURE);
    }

    if((pthread_create(&tid2,NULL,task2,NULL)) != 0){
        fprintf(stderr,"pthread create error\n");
        exit(EXIT_FAILURE);
    }

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


    return 0;
}
//head.h
#ifndef __HEAD_H__
#define __HEAD_H__

#include 
#include 
#include 
#include 
#include 
#include 
#include 

#define PRINT_ERR(msg) do{ \
        perror(msg);\
        return -1;\
    }while(0)

#endif

运行结果:

2.互斥锁的相关函数

头文件均为:#include 

①定义互斥锁
pthread_mutex_t lock;   
    
②初始化互斥锁
pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
//互斥锁的静态初始化

int pthread_mutex_init(pthread_mutex_t * mutex, const pthread_mutexattr_t * attr);
功能:互斥锁的动态初始化
参数:
        @mutex:锁的变量的地址
        @attr:NULL,缺省属性
返回值:成功返回0,失败返回非0
       
③上锁
int pthread_mutex_lock(pthread_mutex_t *mutex);
功能:上锁,如果获取不到锁阻塞等待
参数:
        @mutex:变量的地址
返回值:成功0,失败非0
   
④解锁
int pthread_mutex_unlock(pthread_mutex_t *mutex);
功能:解锁
参数:
        @mutex:变量的地址
返回值:成功0,失败非0
    
⑤销毁锁
int pthread_mutex_destroy(pthread_mutex_t *mutex);
功能:销毁锁
参数:
        @mutex:变量的地址
返回值:成功0,失败非0 

示例:

#include 
#include 
int money=10000;
//定义锁
pthread_mutex_t lock;
void *task1(void *arg)
{
    int get=50;
    while(1){
        pthread_mutex_lock(&lock);
       // printf("我是张三我正在取钱50\n");
        usleep(10000);
        money = money - get;

        if(money > 0){
            printf("张三取钱成功了,账户剩余金额 = %d\n",money);
        }else{
            printf("取钱失败,退出...\n");
        	pthread_mutex_unlock(&lock);
            break;
        }
        pthread_mutex_unlock(&lock);
    }
    pthread_exit(NULL);
}
void *task2(void *arg)
{
    int get=100;
    while(1){
         pthread_mutex_lock(&lock);
       // printf("我是李四我正在取钱100\n");
        usleep(10000);
        money = money - get;

        if(money > 0){
            printf("李四取钱成功了,账户剩余金额 = %d\n",money);
            
        }else{
            printf("取钱失败,退出...\n");
        	  pthread_mutex_unlock(&lock);
            break;
        }
        pthread_mutex_unlock(&lock);
    }
    pthread_exit(NULL);
}
int main(int argc, char const *argv[])
{
    pthread_t tid1,tid2;
    //初始化锁
    pthread_mutex_init(&lock,NULL);

    if((pthread_create(&tid1,NULL,task1,NULL)) != 0){
        fprintf(stderr,"pthread create error\n");
        exit(EXIT_FAILURE);
    }

    if((pthread_create(&tid2,NULL,task2,NULL)) != 0){
        fprintf(stderr,"pthread create error\n");
        exit(EXIT_FAILURE);
    }

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

    //销毁锁
    pthread_mutex_destroy(&lock);
    return 0;
}
3.线程无名信号量(同步)

线程互斥锁解决的是互斥问题,但是多线程没有执行的先后顺序。


但是在实际的问题中又会出现需要控制线程执行先后顺序的问题,比如生产者消费者模型,需要生成者线程先执行,消费者线程后执行,此时互斥锁就解决不了这种问题了,需要使用无名信号量或条件变量来解决。


4.无名信号量的相关函数

头文件均为:#include 

①定义无名信号量
sem_t sem;

②初始化信号
int sem_init(sem_t *sem, int pshared, unsigned int value);
功能:初始化信号量
参数:
        @sem:结构体指针
        @pshared:
                0:多线程间
                1:进程间
        @value:信号量的值,如果填写为1,能获取到信号量,如果是0获取不到信号量
返回值:成功返回0,失败返回-1置位错误码
            
③申请资源
int sem_wait(sem_t *sem);
功能:申请资源,(P *** 作 -1),如果申请不到资源就阻塞等待
参数:
        @sem:结构体指针  
返回值:成功返回0,失败返回-1置位错误码
       
④释放资源
int sem_post(sem_t *sem);
功能:释放资源,(V *** 作 +1)
参数:
        @sem:结构体指针  
返回值:成功返回0,失败返回-1置位错误码  

⑤销毁信号量
int sem_destroy(sem_t *sem);
功能:销毁信号量
参数:
        @sem:结构体指针  
返回值:成功返回0,失败返回-1置位错误码 

示例:生产者&消费者

#include 
#include 
#include 
// 1.定义信号量
sem_t sem1, sem2;

//生产者线程
void *task1(void *arg)
{
    while (1)
    {
        sem_wait(&sem2);
        printf("我生产好了一辆BYD汽车\n");
        sleep(1);
        sem_post(&sem1);
    }
    pthread_exit(NULL);
}
//消费者线程
void *task2(void *arg)
{
    while (1)
    {
        sem_wait(&sem1);
        printf("我购买了一辆BYD汽车\n");
        sleep(1);
        sem_post(&sem2);
    }
    pthread_exit(NULL);
}
int main(int argc, char const *argv[])
{
    pthread_t tid1, tid2;
    // 2.初始化信号量
    sem_init(&sem1, 0, 0);
    sem_init(&sem2, 0, 1);
    if ((pthread_create(&tid1, NULL, task1, NULL)) != 0)
    {
        fprintf(stderr, "pthread create error\n");
        exit(EXIT_FAILURE);
    }

    if ((pthread_create(&tid2, NULL, task2, NULL)) != 0)
    {
        fprintf(stderr, "pthread create error\n");
        exit(EXIT_FAILURE);
    }

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

    // 3.销毁信号量
    sem_destroy(&sem1);
    sem_destroy(&sem2);

    return 0;
}
 5.线程条件变量(同步)

linux内核中可以通过无名信号量解决线程间同步的问题,但是如果线程比较多就需要在线程中定义同等多的信号量,所以这种方式比较繁琐。


内核开发出来了条件变量用于解决这种问题。


6.条件变量的相关函数

头文件均为:#include 

①定义条件变量
 pthread_cond_t cond;

②初始化条件变量
pthread_cond_t cond = PTHREAD_COND_INITIALIZER;
//静态初始化 

int pthread_cond_init(pthread_cond_t *restrict cond,
                                 const pthread_condattr_t *restrict attr);
功能:动态初始化一个条件变量
参数:
        @cond:条件变量的指针
        @attr:NULL使用默认属性
返回值:成功返回0,失败返回非0
       
③阻塞等待条件变量
int pthread_cond_wait(pthread_cond_t *restrict cond,
                                   pthread_mutex_t *restrict mutex);
功能:阻塞等待条件变量,在条件变量中维护了一个队列,这里的互斥锁就是为了解决在往队列中放线程的时候出现竞态问题的。


使用的步骤:
    1.使用pthread_mutex_lock上锁
    2.调用pthread_cond_wait
     2.1将当前线程放入队列
     2.2解锁
     2.3休眠
     2.4获取锁
     2.5休眠状态退出
    3.你的程序
    4.使用pthread_mutex_unlock解锁

参数:
        @cond:条件变量的地址
        @mutex:互斥锁
返回值:成功返回0,失败返回非零
        
     
④给休眠的线程发信号或者广播
int pthread_cond_signal(pthread_cond_t *cond);
功能:唤醒一个休眠的线程
参数:
        @cond:条件变量的地址
返回值:成功返回0,失败返回非零

int pthread_cond_broadcast(pthread_cond_t *cond);
功能:唤醒所有休眠的线程
参数:
        @cond:条件变量的地址
返回值:成功返回0,失败返回非零     
       
⑤销毁条件变量     
int pthread_cond_destroy(pthread_cond_t *cond);
功能:销毁条件变量
参数:
        @cond:条件变量的地址
返回值:成功返回0,失败返回非零

示例:唤醒多个休眠的线程

#include 
#include 
#include 
// 1.定义条件变量
pthread_cond_t cond;
pthread_mutex_t mutex;
//生产者线程
void *task1(void *arg)
{
    int num=10;
    while (--num)
    {
        printf("我生产好了4辆BYD汽车\n");
        sleep(3);
        //pthread_cond_signal(&cond);  //唤醒一个休眠的线程
        pthread_cond_broadcast(&cond); //唤醒所有的休眠的线程
    }
    pthread_exit(NULL);
}
//消费者线程
void *task2(void *arg)
{
    pthread_mutex_lock(&mutex);
    pthread_cond_wait(&cond, &mutex);
    printf("我购买了一辆BYD汽车,tid = %ld\n",pthread_self());
    pthread_mutex_unlock(&mutex);
    pthread_exit(NULL);
}
int main(int argc, char const *argv[])
{
    pthread_t tid1, tid2, tid3, tid4, tid5;
    // 2初始化条件变量
    pthread_cond_init(&cond, NULL);
    pthread_mutex_init(&mutex, NULL);
    if ((pthread_create(&tid1, NULL, task1, NULL)) != 0)
    {
        fprintf(stderr, "pthread create error\n");
        exit(EXIT_FAILURE);
    }

    if ((pthread_create(&tid2, NULL, task2, NULL)) != 0)
    {
        fprintf(stderr, "pthread create error\n");
        exit(EXIT_FAILURE);
    }
    if ((pthread_create(&tid3, NULL, task2, NULL)) != 0)
    {
        fprintf(stderr, "pthread create error\n");
        exit(EXIT_FAILURE);
    }
    if ((pthread_create(&tid4, NULL, task2, NULL)) != 0)
    {
        fprintf(stderr, "pthread create error\n");
        exit(EXIT_FAILURE);
    }
    if ((pthread_create(&tid5, NULL, task2, NULL)) != 0)
    {
        fprintf(stderr, "pthread create error\n");
        exit(EXIT_FAILURE);
    }
    printf("tid2 = %ld\n",tid2);
    printf("tid3 = %ld\n",tid3);
    printf("tid4 = %ld\n",tid4);
    printf("tid5 = %ld\n",tid5);

    pthread_join(tid1, NULL);
    pthread_join(tid2, NULL);
    pthread_join(tid3, NULL);
    pthread_join(tid4, NULL);
    pthread_join(tid5, NULL);
    // 3.销毁条件变量
    pthread_cond_destroy(&cond);
    pthread_mutex_destroy(&mutex);

    return 0;
}

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/langs/567828.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-04-09
下一篇 2022-04-09

发表评论

登录后才能评论

评论列表(0条)

保存