当前位置: 首页 > news >正文

lv5 嵌入式开发-6 线程的取消和互斥

目录

1 线程通信 – 互斥

2 互斥锁初始化 – pthread_mutex_init

3 互斥锁销毁 pthread_mutex_destroy

4 申请锁 – pthread_mutex_lock

5 释放锁 – pthread_mutex_unlock

6 读写锁

7 死锁的避免

8 条件变量(信号量)

9 线程池概念和实现

9.1 概念

9.2 线程池的实现

9.3 练习

10 线程的GDB调试


掌握:临界资源(了解)、互斥机制(理解)、互斥锁(熟练)

1 线程通信 – 互斥

临界资源  一次只允许一个任务(进程、线程)访问的共享资源

临界区 访问临界资源的代码

互斥机制 mutex互斥锁 任务访问临界资源前申请锁,访问完后释放锁

2 互斥锁初始化 – pthread_mutex_init

两种方法创建互斥锁,静态方式动态方式

动态方式:

#include  <pthread.h>int  pthread_mutex_init(pthread_mutex_t *mutex,const pthread_mutexattr_t *  attr);
  • 成功时返回0,失败时返回错误码  
  • mutex  指向要初始化的互斥锁对象  
  • attr  互斥锁属性,NULL表示缺省属性
  • man 函数出现 No manual entry for pthread_mutex_xxx   -解决办法     apt-get install manpages-posix-dev

静态方式:

pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

3 互斥锁销毁 pthread_mutex_destroy

int pthread_mutex_destroy(pthread_mutex_t *mutex)

4 申请锁 – pthread_mutex_lock

 #include  <pthread.h>int  pthread_mutex_lock(pthread_mutex_t *mutex);int pthread_mutex_trylock(pthread_mutex_t *mutex)
  • 成功时返回0,失败时返回错误码  
  • mutex  指向要初始化的互斥锁对象  
  • pthread_mutex_lock 如果无法获得锁,任务阻塞
  • pthread_mutex_trylock 如果无法获得锁,返回EBUSY而不是挂起等待

5 释放锁 – pthread_mutex_unlock

 #include  <pthread.h>int  pthread_mutex_unlock(pthread_mutex_t *mutex);
  • 成功时返回0,失败时返回错误码  
  • mutex  指向要初始化的互斥锁对象  
  • 执行完临界区要及时释放锁

示例:两个线程同时写一个文件的现象。

#include <stdio.h>
#include <pthread.h>
#include <unistd.h>
#include <string.h>FILE *fp;void *func2(void *arg)
{pthread_detach(pthread_self());char c;int i=0;printf("This is func2 thread\n");char str[]="I write func2 line";while(1){while(i<strlen(str)){c = str[i];fputc(c,fp);usleep(10);i++;}i = 0;usleep(1);}pthread_exit("func2 return");
}void *func(void *arg)
{pthread_detach(pthread_self());int i =0;char c;printf("This is func1 thread\n");char str[]="You read func1 thread\n";while(1){while(i<strlen(str)){c = str[i];fputc(c,fp);usleep(10);i++;}i = 0;usleep(1);}pthread_exit("func1 return");
}int main(int argc,char * argv[])
{pthread_t tid,tid2;void *retv;fp = fopen("1.txt","a+");if(fp == NULL){perror("fopen");return 0;}pthread_create(&tid,NULL,func,NULL);pthread_create(&tid2,NULL,func2,NULL);while(1){sleep(1);}
}//实验现象Yfuounc r2 ealid nefunI c1wr tithre eafud
nc2Yo uli nreeaId  wfurincte1  tfuhrncea2 d
liYneouI r weardit efu fncu1nc t2 hrlieaned
IY wouri rteea fd unfuc2nc 1li tnehreI adw
riYteou f runeac2d  lfuinnce1 I thwrreiatdefYunouc2 r leaidne fIu ncwr1 itthe refuadn
c2Y loiu nreeIad w friuntec1  futhncre2 adli
neYoIu  wreriadte f fununc1c2 t lhrineaed
I Ywrouit re eafudn fc2un lc1in tehIre wadri
teYo fu unrec2ad l finunec1I  twrhriteae d
fuYncou2  rlieaned Ifu wncri1 teth freunadc2lYoinu ereI adwr fitune c1fu tnchr2 ealidn
eYoI u wrreitade  ffuunncc12  tlihrneeaIdwYrioute r feaund cf2 unlic1ne tIhr wearid
te Yofuun rc2ea ld infuencI1  wtrhirteea dfu

加上互斥锁的示例:

#include <stdio.h>
#include <pthread.h>
#include <unistd.h>
#include <string.h>pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;  //初始化
FILE *fp;void *func2(void *arg)
{pthread_detach(pthread_self());char c;int i=0;printf("This is func2 thread\n");char str[]="I write func2 line";while(1){pthread_mutex_lock(&mutex);  //加锁while(i<strlen(str)){c = str[i];fputc(c,fp);usleep(10);i++;}pthread_mutex_unlock(&mutex); //解锁i = 0;usleep(1);}pthread_exit("func2 return");
}void *func(void *arg)
{pthread_detach(pthread_self());int i =0;char c;printf("This is func1 thread\n");char str[]="You read func1 thread\n";while(1){pthread_mutex_lock(&mutex);while(i<strlen(str)){c = str[i];fputc(c,fp);usleep(10);i++;}pthread_mutex_unlock(&mutex);i = 0;usleep(1);}pthread_exit("func1 return");
}int main(int argc,char * argv[])
{pthread_t tid,tid2;void *retv;fp = fopen("1.txt","a+");if(fp == NULL){perror("fopen");return 0;}pthread_create(&tid,NULL,func,NULL);pthread_create(&tid2,NULL,func2,NULL);while(1){sleep(1);}
}//结果
I write func2 lineYou read func1 thread
I write func2 lineYou read func1 thread
I write func2 lineYou read func1 thread
I write func2 lineYou read func1 thread
I write func2 lineYou read func1 thread
I write func2 lineYou read func1 thread
I write func2 lineYou read func1 thread
I write func2 lineYou read func1 thread
I write func2 lineYou read func1 thread

6 读写锁

问:多个线程只是读文件,这时候不会造成文件写坏,加了互斥锁会出现什么问题?

如果加了互斥锁,读文件的效率很低,多个线程读文件是不影响的。

如果一个线程在写,多个线程在读,那么读到一半,文件被改了,那么会出现读错误。

读写锁必要性:提高线程执行效率

特性:

写者:写者使用写锁,如果当前没有读者,也没有其他写者,写者立即获得写锁;否则写者将等待,直到没有读者和写者。

读者:读者使用读锁,如果当前没有写者,读者立即获得读锁;否则读者等待,直到没有写者。

注意:

-同一时刻只有一个线程可以获得写锁,同一时刻可以有多个线程获得读锁。

-读写锁出于写锁状态时,所有试图对读写锁加锁的线程,不管是读者试图加读锁,还是写者试图加写锁,都会被阻塞。

-读写锁处于读锁状态时,有写者试图加写锁时,之后的其他线程的读锁请求会被阻塞,以避免写者长时间的不写锁

初始化一个读写锁   pthread_rwlock_init
读锁定读写锁        pthread_rwlock_rdlock
非阻塞读锁定  pthread_rwlock_tryrdlock
写锁定读写锁      pthread_rwlock_wrlock
非阻塞写锁定      pthread_rwlock_trywrlock
解锁读写锁         pthread_rwlock_unlock
释放读写锁         pthread_rwlock_destroy

 示例:

#include <pthread.h>
#include <stdio.h>
#include <unistd.h>
#include <string.h>pthread_rwlock_t rwlock;FILE *fp;
void * read_func(void *arg){pthread_detach(pthread_self());printf("read thread\n");char buf[32]={0};while(1){//rewind(fp);   //从线程开头读pthread_rwlock_rdlock(&rwlock);           //如果加了wrlock,那么线程1 读完才轮到线程2读while(fgets(buf,32,fp)!=NULL){printf("%d,rd=%s\n",(int)arg,buf);usleep(1000);}pthread_rwlock_unlock(&rwlock);sleep(1);}}void *func2(void *arg){pthread_detach(pthread_self());printf("This func2 thread\n");char str[]="I write func2 line\n";char c;int i=0;while(1){pthread_rwlock_wrlock(&rwlock);while(i<strlen(str)){c = str[i];fputc(c,fp);usleep(1);i++;}pthread_rwlock_unlock(&rwlock);i=0;usleep(1);}pthread_exit("func2 exit");}void *func(void *arg){pthread_detach(pthread_self());printf("This is func1 thread\n");char str[]="You read func1 thread\n";char c;int i=0;while(1){pthread_rwlock_wrlock(&rwlock);while(i<strlen(str)){c = str[i];fputc(c,fp);i++;usleep(1);}pthread_rwlock_unlock(&rwlock);i=0;usleep(1);}pthread_exit("func1 exit");
}int main(){pthread_t tid1,tid2,tid3,tid4;void *retv;int i;fp = fopen("1.txt","a+");if(fp==NULL){perror("fopen");return 0;}pthread_rwlock_init(&rwlock,NULL);pthread_create(&tid1,NULL,read_func,1);pthread_create(&tid2,NULL,read_func,2);pthread_create(&tid3,NULL,func,NULL);pthread_create(&tid4,NULL,func2,NULL);while(1){    sleep(1);} }

2个读线程创建的快一点,如果没读完,写线程是写不进去的。如果先写,同样读也要等待。 

7 死锁的避免

什么是死锁

一把锁是不会出现死锁的。一般两把以上才会出现死锁 

 示例:模式死锁

#include <stdio.h>
#include <pthread.h>
#include <unistd.h>
#include <string.h>pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
pthread_mutex_t mutex2 = PTHREAD_MUTEX_INITIALIZER;
FILE *fp;void *func2(void *arg)
{pthread_detach(pthread_self());printf("This is func2 thread\n");while(1){pthread_mutex_lock(&mutex2);printf("%d,I got lock2\n",(int)arg);sleep(1);pthread_mutex_lock(&mutex);printf("%d,I got 2 locks\n",(int)arg);pthread_mutex_unlock(&mutex);pthread_mutex_unlock(&mutex2);usleep(1);}pthread_exit("func2 return");
}void *func(void *arg)
{pthread_detach(pthread_self());printf("This is func1 thread\n");while(1){pthread_mutex_lock(&mutex);printf("%d,I got lock1\n",(int)arg);sleep(1);pthread_mutex_lock(&mutex2);printf("%d,I got 2 locks\n",(int)arg);pthread_mutex_unlock(&mutex2);pthread_mutex_unlock(&mutex);usleep(1);}pthread_exit("func1 return");
}int main(int argc,char * argv[])
{pthread_t tid,tid2;void *retv;fp = fopen("1.txt","a+");if(fp == NULL){perror("fopen");return 0;}pthread_create(&tid,NULL,func,1);pthread_create(&tid2,NULL,func2,2);while(1){sleep(1);}
}//死锁结果
linux@linux:~/Desktop$ gcc -g -o mutex mutex.c -lpthread
mutex.c: In function ‘main’:
mutex.c:61:2: warning: passing argument 4 of ‘pthread_create’ makes pointer from integer without a cast [enabled by default]pthread_create(&tid,NULL,func,1);^
In file included from mutex.c:2:0:
/usr/include/pthread.h:244:12: note: expected ‘void * __restrict__’ but argument is of type ‘int’extern int pthread_create (pthread_t *__restrict __newthread,^
mutex.c:62:2: warning: passing argument 4 of ‘pthread_create’ makes pointer from integer without a cast [enabled by default]pthread_create(&tid2,NULL,func2,2);^
In file included from mutex.c:2:0:
/usr/include/pthread.h:244:12: note: expected ‘void * __restrict__’ but argument is of type ‘int’extern int pthread_create (pthread_t *__restrict __newthread,^
linux@linux:~/Desktop$ ./mutex 
This is func2 thread
2,I got lock2
This is func1 thread
1,I got lock1。。。

解决方法1:通过时间差让线程1 先执行,获取两把锁后再休息,线程2再执行,获取两把锁后再休息

#include <stdio.h>
#include <pthread.h>
#include <unistd.h>
#include <string.h>pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
pthread_mutex_t mutex2 = PTHREAD_MUTEX_INITIALIZER;
FILE *fp;void *func2(void *arg)
{pthread_detach(pthread_self());printf("This is func2 thread\n");while(1){pthread_mutex_lock(&mutex2);printf("%d,I got lock2\n",(int)arg);sleep(1);pthread_mutex_lock(&mutex);printf("%d,I got 2 locks\n",(int)arg);pthread_mutex_unlock(&mutex);pthread_mutex_unlock(&mutex2);sleep(5);}pthread_exit("func2 return");
}void *func(void *arg)
{pthread_detach(pthread_self());printf("This is func1 thread\n");while(1){pthread_mutex_lock(&mutex);printf("%d,I got lock1\n",(int)arg);sleep(1);pthread_mutex_lock(&mutex2);printf("%d,I got 2 locks\n",(int)arg);pthread_mutex_unlock(&mutex2);pthread_mutex_unlock(&mutex);sleep(5);}pthread_exit("func1 return");
}int main(int argc,char * argv[])
{pthread_t tid,tid2;void *retv;fp = fopen("1.txt","a+");if(fp == NULL){perror("fopen");return 0;}pthread_create(&tid,NULL,func,1);sleep(2);pthread_create(&tid2,NULL,func2,2);while(1){sleep(1);}
}//执行结果
linux@linux:~/Desktop$ ./mutex 
This is func1 thread
1,I got lock1
1,I got 2 locks
This is func2 thread
2,I got lock2
2,I got 2 locks
1,I got lock1
1,I got 2 locks
2,I got lock2
1,I got lock1

解决方法2:调整锁的顺序。都先获取锁1,再去获取锁2,不会同时造成2个资源被锁的情况。

#include <stdio.h>
#include <pthread.h>
#include <unistd.h>
#include <string.h>pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
pthread_mutex_t mutex2 = PTHREAD_MUTEX_INITIALIZER;
FILE *fp;void *func2(void *arg)
{pthread_detach(pthread_self());printf("This is func2 thread\n");while(1){pthread_mutex_lock(&mutex);printf("%d,I got lock2\n",(int)arg);sleep(1);pthread_mutex_lock(&mutex2);printf("%d,I got 2 locks\n",(int)arg);pthread_mutex_unlock(&mutex2);pthread_mutex_unlock(&mutex);sleep(1);}pthread_exit("func2 return");
}void *func(void *arg)
{pthread_detach(pthread_self());printf("This is func1 thread\n");while(1){pthread_mutex_lock(&mutex);printf("%d,I got lock1\n",(int)arg);sleep(1);pthread_mutex_lock(&mutex2);printf("%d,I got 2 locks\n",(int)arg);pthread_mutex_unlock(&mutex2);pthread_mutex_unlock(&mutex);sleep(1);}pthread_exit("func1 return");
}int main(int argc,char * argv[])
{pthread_t tid,tid2;void *retv;fp = fopen("1.txt","a+");if(fp == NULL){perror("fopen");return 0;}pthread_create(&tid,NULL,func,1);pthread_create(&tid2,NULL,func2,2);while(1){sleep(1);}
}linux@linux:~/Desktop$ ./mutex 
This is func2 thread
2,I got lock2
This is func1 thread
2,I got 2 locks
1,I got lock1
1,I got 2 locks
2,I got lock2
2,I got 2 locks
1,I got lock1
1,I got 2 locks
2,I got lock2
2,I got 2 locks
1,I got lock1
1,I got 2 locks
2,I got lock2

总结:

  1. 锁越少越好,最好使用一把锁
  2. 调整好锁的顺序

 练习:实现多个线程写一个文件,使用互斥锁

#include <stdio.h>
#include <pthread.h>
#include <unistd.h>
#include <string.h>pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;void *write_func1(void *arg)
{pthread_detach(pthread_self());printf("This is write_func1 thread\n");while(1){pthread_mutex_lock(&mutex);printf("%d,I got lock\n",(int)arg);pthread_mutex_unlock(&mutex);sleep(1);}pthread_exit("write_func1 return");
}void *write_func2(void *arg)
{pthread_detach(pthread_self());printf("This is write_func2 thread\n");while(1){pthread_mutex_lock(&mutex);printf("%d,I got lock\n",(int)arg);pthread_mutex_unlock(&mutex);sleep(1);}pthread_exit("write_func2 return");
}int main(int argc,char * argv[])
{pthread_t tid,tid2;void *retv;pthread_create(&tid,NULL,write_func1,1);pthread_create(&tid2,NULL,write_func2,2);while(1){sleep(1);}
}

8 条件变量(信号量)

应用场景:生产者消费者问题,是线程同步的一种手段。

必要性:为了实现等待某个资源,让线程休眠。提高运行效率

pthread_cond_wait(&m_cond,&m_mutex);                //完全阻塞等待int pthread_cond_timedwait(pthread_cond_t *restrict cond,  //超时等待pthread_mutex_t *restrict mutex, const struct timespec *restrict abstime);int pthread_cond_signal(pthread_cond_t *cond);            //通知1个线程
int pthread_cond_broadcast(pthread_cond_t *cond);         //通知多个线程

使用方法:

静态初始化或使用动态初始化

pthread_cond_t   cond = PTHREAD_COND_INITIALIZER;      //静态初始化条件变量
pthread_mutex_t  mutex = PTHREAD_MUTEX_INITIALIZER;    //初始化互斥量pthread_cond_t cond;                                   //动态初始化条件变量
pthread_cond_init(&cond);                              //动态初始化条件变量

  生产资源线程:

pthread_mutex_lock(&mutex);开始产生资源pthread_cond_sigal(&cond);    //通知一个消费线程或者
pthread_cond_broadcast(&cond); //广播通知多个消费线程pthread_mutex_unlock(&mutex);

消费者线程:

pthread_mutex_lock(&mutex);while (如果没有资源){   //防止惊群效应pthread_cond_wait(&cond, &mutex); }有资源了,消费资源pthread_mutex_unlock(&mutex);  

示例:

#include <pthread.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>pthread_cond_t  hasTaxi=PTHREAD_COND_INITIALIZER;
pthread_mutex_t lock  = PTHREAD_MUTEX_INITIALIZER;struct taxi{struct taxi *next;int num;};struct taxi *Head=NULL;void *taxiarv(void *arg){printf("taxi arrived thread\n");pthread_detach(pthread_self());struct taxi *tx;int i=1;while(1){tx = malloc(sizeof(struct taxi));tx->num = i++;printf("taxi %d comming\n",tx->num);pthread_mutex_lock(&lock);tx->next = Head;Head = tx;pthread_cond_signal(&hasTaxi);  //生产了一个资源信号//pthread_cond_broadcast(&hasTaxi);  //有可能产生段错误pthread_mutex_unlock(&lock);sleep(1);}pthread_exit(0);
}void *takeTaxi(void *arg){printf("take taxi thread\n");pthread_detach(pthread_self());struct taxi *tx;while(1){pthread_mutex_lock(&lock);while(Head==NULL)  //这句不能去{pthread_cond_wait(&hasTaxi,&lock);}//有资源了可以消费tx = Head;Head=tx->next;printf("%d,Take taxi %d\n",(int)arg,tx->num);free(tx);pthread_mutex_unlock(&lock);}pthread_exit(0);
}int main(){pthread_t tid1,tid2,tid3;pthread_create(&tid1,NULL,taxiarv,NULL);
//    sleep(5);pthread_create(&tid2,NULL,takeTaxi,(void*)1);  //(谁先获得信号谁执行,没有先后规律,并行。pthread_create(&tid2,NULL,takeTaxi,(void*)2);pthread_create(&tid2,NULL,takeTaxi,(void*)3);while(1) {sleep(1);}}

注意:

1 pthread_cond_wait(&cond, &mutex)在没有资源等待是是先unlock 休眠,等资源到了,再lock

所以pthread_cond_wait he pthread_mutex_lock 必须配对使用

2  如果pthread_cond_signal或者pthread_cond_broadcast 早于 pthread_cond_wait ,则有可能会丢失信号。(对应代码中 while(Head==NULL)  不能去)

3 pthead_cond_broadcast 信号会被多个线程收到,这叫线程的惊群效应。所以需要加上判断条件while循环。(需要加上while(Head==NULL)  防止同时获取,空指针被获取)

练习:

条件变量有两种初始化的方式,写出这两种方式:

pthread_cond_t cond = PTHREAD_COND_INITIALIZER;      //静态初始化条件变量pthread_cond_t cond;                                        //动态初始化条件变量
pthread_cond_init(&cond, NULL);

9 线程池概念和实现

9.1 概念

概念:

通俗的讲就是一个线程的池子,可以循环的完成任务的一组线程集合

必要性:

我们平时创建一个线程,完成某一个任务,等待线程的退出。但当需要创建大量的线程时,假设T1创建线程时间,T2在线程任务执行时间,T3线程销毁时间 T1+T3 > T2这时候就不划算了,使用线程池可以降低频繁创建和销毁线程所带来的开销,任务处理时间比较短的时候这个好处非常显著

线程池的基本结构:

1 任务队列,存储需要处理的任务,由工作线程来处理这些任务

2 线程池工作线程,它是任务队列任务的消费者,等待新任务的信号

9.2 线程池的实现

创建线程池的基本结构:

任务队列链表  typedef struct Task;

线程池结构体  typedef struct ThreadPool;

线程池的初始化:

pool_init()
{创建一个线程池结构实现任务队列互斥锁和条件变量的初始化创建n个工作线程
}

线程池添加任务

pool_add_task
{判断是否有空闲的工作线程给任务队列添加一个节点给工作线程发送信号newtask
}

实现工作线程

workThread
{while(1){等待newtask任务信号从任务队列中删除节点执行任务}
}

线程池的销毁

pool_destory
{删除任务队列链表所有节点,释放空间删除所有的互斥锁条件变量删除线程池,释放空间
}

示例:

#include <pthread.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>#define POOL_NUM 10
typedef struct Task{void *(*func)(void *arg);void *arg;struct Task *next;
}Task;typedef struct ThreadPool{pthread_mutex_t taskLock;pthread_cond_t newTask;pthread_t tid[POOL_NUM];Task *queue_head;int busywork;}ThreadPool;ThreadPool *pool;void *workThread(void *arg){while(1){pthread_mutex_lock(&pool->taskLock);pthread_cond_wait(&pool->newTask,&pool->taskLock);Task *ptask = pool->queue_head;pool->queue_head = pool->queue_head->next;pthread_mutex_unlock(&pool->taskLock);ptask->func(ptask->arg);pool->busywork--;}
}void *realwork(void *arg){printf("Finish work %d\n",(int)arg);
}void pool_add_task(int arg){Task *newTask;pthread_mutex_lock(&pool->taskLock);while(pool->busywork>=POOL_NUM){pthread_mutex_unlock(&pool->taskLock);  //休眠时候释放锁usleep(10000);              //线程池满等待pthread_mutex_lock(&pool->taskLock);     //休眠结束再锁,否则别人访问不到资源}pthread_mutex_unlock(&pool->taskLock);newTask = malloc(sizeof(Task));newTask->func =  realwork;newTask->arg = arg;pthread_mutex_lock(&pool->taskLock);  //操作队列需要加锁Task *member = pool->queue_head;if(member==NULL){pool->queue_head = newTask;}else{while(member->next!=NULL){          //新任务插入队列尾部member=member->next;}member->next = newTask;}pool->busywork++;pthread_cond_signal(&pool->newTask);pthread_mutex_unlock(&pool->taskLock);
}void pool_init(){pool = malloc(sizeof(ThreadPool));pthread_mutex_init(&pool->taskLock,NULL);pthread_cond_init(&pool->newTask,NULL);pool->queue_head = NULL;pool->busywork=0;for(int i=0;i<POOL_NUM;i++){pthread_create(&pool->tid[i],NULL,workThread,NULL);}
}void pool_destory(){Task *head;while(pool->queue_head!=NULL){head = pool->queue_head;pool->queue_head = pool->queue_head->next;free(head);}pthread_mutex_destroy(&pool->taskLock);pthread_cond_destroy(&pool->newTask);free(pool);
}int main(){pool_init();sleep(20);for(int i=1;i<=20;i++){pool_add_task(i);}sleep(5);pool_destory();}

编译错误:

error: ‘ThreadPool {aka struct ThreadPool}’ has no member named ‘head’

意义:ThreadPool 结构体没有head这个成员。

解决:检查是否拼写错误。

error: too few arguments to function ‘pthread_mutex_init’

意思:pthread_mutex_init这个函数参数少了

解决:检查函数的参数,添加对应的参数

运行结果:20个任务共享10个线程池,不让任务丢失。

9.3 练习

实现课程线程池代码

#include <stdio.h>
#include <pthread.h>
#include <stdlib.h>#define POOL_MAX_NUM 10typedef struct _Task
{void *(*func)(void *arg);void *arg;struct _Task * next;
}Task;typedef struct _ThreadPool
{pthread_mutex_t taskLock;pthread_cond_t newTask;pthread_t tid[POOL_MAX_NUM];Task *queue_head;int busywork;
}ThreadPool;ThreadPool *pool;void *realwork(void *arg)
{printf("Finish work %d\n",(int)arg);
}void pool_add_task(int arg)
{Task *newTask;pthread_mutex_lock(&pool->taskLock); while(pool->busywork >= POOL_MAX_NUM){pthread_mutex_unlock(&pool->taskLock);usleep(10000);pthread_mutex_lock(&pool->taskLock);}pthread_mutex_unlock(&pool->taskLock);newTask = malloc(sizeof(Task));newTask->func = realwork;newTask->arg = &arg;pthread_mutex_lock(&pool->taskLock);Task * member = pool->queue_head;if(member == NULL){pool->queue_head = newTask;}else{while(member->next != NULL){member = member->next;}member->next = newTask;}pool->busywork ++;pthread_cond_signal(&pool->newTask);pthread_mutex_unlock(&pool->taskLock);}void *workThread(void *arg)
{while(1){pthread_mutex_lock(&pool->taskLock);//等待newtask任务信号pthread_cond_wait(&pool->newTask,&pool->taskLock);//从队列中删除一个节点Task *ptask = pool->queue_head;pool->queue_head = pool->queue_head->next;pthread_mutex_unlock(&pool->taskLock);//执行任务ptask->func(ptask->arg);pool->busywork--;}
}void pool_init()
{int i;pool = malloc(sizeof(ThreadPool));pthread_mutex_init(&pool->taskLock,NULL);pthread_cond_init(&pool->newTask,NULL);pool->queue_head = NULL;pool->busywork = 0;for(i = 0; i < POOL_MAX_NUM; i++){pthread_create(&pool->tid[i],NULL,workThread,NULL);}
}void pool_destory(){Task *head;while(pool->queue_head!=NULL){head = pool->queue_head;pool->queue_head = pool->queue_head->next;free(head);}pthread_mutex_destroy(&pool->taskLock);pthread_cond_destroy(&pool->newTask);free(pool);
}int main(int argc,char *argv[])
{int i;pool_init();sleep(5);for(i = 1; i <= 30; i++){pool_add_task(i);}sleep(5);pool_destory();}

10 线程的GDB调试

显示线程

info thread

切换线程

thread xxx
b 6 thread 3  //线程运行后给线程3第6行打端点
bt  //可以打印出当前线程的函数调用栈信息。它会显示函数调用链的序列,从当前执行点一直追溯到代码的起始点,以帮助开发人员定位问题所在

GDB为特定线程设置断点

break location thread id

GDB设置线程锁

set scheduler-locking on/off  //on:其他线程会暂停。可以单独调试一个线程

#include <pthread.h>
#include <stdio.h>void *testThread(void *arg){char *threadName = (char*)arg;printf("Current running %s\n",threadName);printf("aaaaaaaa\n");printf("bbbbbbbb\n");pthread_exit(0);}int main(){pthread_t tid1,tid2;pthread_create(&tid1,NULL,testThread,"thread1");pthread_create(&tid2,NULL,testThread,"thread2");pthread_join(tid1,NULL);pthread_join(tid2,NULL);}

相关文章:

lv5 嵌入式开发-6 线程的取消和互斥

目录 1 线程通信 – 互斥 2 互斥锁初始化 – pthread_mutex_init 3 互斥锁销毁 pthread_mutex_destroy 4 申请锁 – pthread_mutex_lock 5 释放锁 – pthread_mutex_unlock 6 读写锁 7 死锁的避免 8 条件变量&#xff08;信号量&#xff09; 9 线程池概念和实现 9.1 …...

el-table实现穿梭功能

第一种 <template><el-row :gutter"20"><el-col :span"10"><!-- 搜索 --><div class"search-bg"><YcSearchInput title"手机号" v-model"search.phone" /><div class"search-s…...

Cron表达式_用于定时调度任务

一、Cron表达式简介 Cron表达式是一个用于设置计划任务的字符串&#xff0c;该字符串以5或6个空格分隔&#xff0c;分为6或7个域&#xff0c;每一个域代表任务在相应时间、日期或时间间隔执行的规则【Cron表达式最初是在类Unix操作中系统中使用的&#xff0c;但现在已经广泛应用…...

CeresPCL ICP精配准(点到面)

文章目录 一、简介二、实现代码三、实现效果参考资料一、简介 ICP算法总共分为6个阶段,如下图所示: (1)挑选发生重叠的点云子集,这一步如果原始点云数据量比较巨大,一般会对原始点云进行下采样操作。 (2)匹配特征点。通常是距离最近的两个点,当然这需要视评判的准则而…...

CentOS安装kafka单机部署

一&#xff1a;保证机器上已经运行的有Java环境 服务器&#xff1a;centos7 kafka版本&#xff1a;3.5.1 二&#xff1a;下载kafka压缩包 下载地址 1.解压kafka压缩包 tar -zxvf kafka_2.13-3.5.1.tgz 2.我得是上传到了 /home目录下&#xff0c;配置文件server.propertie…...

蓝牙核心规范(V5.4)11.2-LE Audio 笔记之LE Auido架构

专栏汇总网址&#xff1a;蓝牙篇之蓝牙核心规范学习笔记&#xff08;V5.4&#xff09;汇总_蓝牙核心规范中文版_心跳包的博客-CSDN博客 爬虫网站无德&#xff0c;任何非CSDN看到的这篇文章都是盗版网站&#xff0c;你也看不全。认准原始网址。&#xff01;&#xff01;&#x…...

福建江夏学院蔡慧梅主任一行莅临拓世科技集团,共探AI+时代教育新未来

在科技的海洋中&#xff0c;产业是那航行的巨轮&#xff0c;而教育则是指引方向的灯塔。当巨轮与灯塔相互辉映&#xff0c;产教融合与校企合作便成为了推动国家科技创新和人才培养的金钥匙&#xff0c;为未来开启一扇扇充满希望的大门。 2023年9月24日&#xff0c;福建江夏学院…...

使用Visual Leak Detector排查内存泄漏问题

目录 1、VLD工具概述 2、下载、安装VLD 2.1、下载VLD 2.2、安装VLD 3、VLD安装目录及文件说明 3.1、安装目录及文件说明 3.2、关于32位和64位版本的详细说明 4、在工程中引入VLD 5、内存泄漏检测实例讲解 5.1、程序启动报错 5.2、启动调试&#xff0c;查看内存泄漏报…...

如何设计一个 JVM 语言下的 LLM 应用开发框架?以 Chocolate Factory 为例

本文将介绍 Chocolate Factory 框架背后的一系列想法和思路。在我们探索和设计框架的过程中&#xff0c;受到了&#xff1a;LangChain4j、LangChain、LlamaIndex、Spring AI、Semantic Kernel、PromptFlow 的大量启发。 欢迎一起来探索&#xff1a;https://github.com/unit-mes…...

基础排序算法

插入排序&#xff08;insertion sort&#xff09; 插入排序每次循环将一个元素放置在适当的位置。像抓牌一样。手里的排是有序的&#xff0c;新拿一张牌&#xff0c;与手里的牌进行比较将其放在合适的位置。 插入排序要将待排序的数据分成两部分&#xff0c;一部分有序&#…...

Nginx的反向代理、动静分离、负载均衡

反向代理 反向代理是一种常见的网络技术&#xff0c;它可以将客户端的请求转发到服务器群集中的一个或多个后端服务器上进行处理&#xff0c;并将响应结果返回给客户端。反向代理技术通常用于提高网站的可伸缩性和可用性&#xff0c;并且可以隐藏真实的后端服务器地址。 #user…...

LLM-TAP随笔——大语言模型基础【深度学习】【PyTorch】【LLM】

文章目录 2.大语言模型基础2.1、编码器和解码器架构2.2、注意力机制2.2.1、注意力机制&#xff08;Attention&#xff09;2.2.2、自注意力机制&#xff08;Self-attention&#xff09;2.2.3、多头自注意力&#xff08;Multi-headed Self-attention&#xff09; 2.3、transforme…...

蓝桥杯备赛-上学迟到

上学迟到 P5707 【深基2.例12】上学迟到 - 洛谷 |https://www.luogu.com.cn/problem/P5707 题目介绍 题目描述 学校和 yyy 的家之间的距离为 s 米&#xff0c;而 yyy 以v 米每分钟的速度匀速走向学校。 在上学的路上&#xff0c;yyy 还要额外花费 1010 分钟的时间进行垃圾分…...

基于 MATLAB 的电力系统动态分析研究【IEEE9、IEEE68系节点】

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…...

2023百度之星 题目详解 公园+糖果促销

2023百度之星题目详解 文章目录 2023百度之星题目详解前言公园问题题目详解 夏日漫步问题问题详情题目详解 前言 这里为大家带来最新的2023百度之星的题目详解&#xff0c;后续还会继续更新&#xff0c;喜欢的小伙伴可以点个关注啦&#xff01; 公园问题 今天是六一节&#…...

C++ 2019-2022 CSP_J 复赛试题横向维度分析(中)

上文讲解了2019~2022年第一题和第二题。第一题偏数学认知&#xff0c;算法较简单&#xff0c;第二题考查基本数据结构&#xff0c;如队列、栈……和基础算法&#xff0c;如排序、模拟……。 本文继续讲解第三题和第四题。 1. 第三题 1.1 2022 题目&#xff1a; 逻辑表达式…...

基于Spring Boot的IT技术交流和分享平台的设计与实现

目录 前言 一、技术栈 二、系统功能介绍 三、核心代码 1、登录模块 2、文件上传模块 3、代码封装 前言 我国科学技术的不断发展&#xff0c;计算机的应用日渐成熟&#xff0c;其强大的功能给人们留下深刻的印象&#xff0c;它已经应用到了人类社会的各个层次的领域&#x…...

智算引领·创新未来 | 2023紫光展锐泛物联网终端生态论坛成功举办

9月21日&#xff0c;紫光展锐在深圳成功举办2023泛物联网终端生态论坛。论坛以“智算引领创新未来”为主题&#xff0c;吸引了来自信通院、中国联通、中国移动、中国电信、金融机构、终端厂商、模组厂商等行业各领域三百多位精英翘楚汇聚一堂&#xff0c;探讨在连接、算力驱动下…...

网络安全技术指南 103.91.209.X

网络安全技术指的是一系列防范网络攻击、保护网络安全的技术手段和措施&#xff0c;旨在保护网络的机密性、完整性和可用性。常见的网络安全技术包括&#xff1a; 防火墙&#xff1a;用于监控网络流量&#xff0c;过滤掉可能包括恶意软件的数据包。 加密技术&#xff1a;用于保…...

用flex实现grid布局

1. css代码 .flexColumn(columns, gutterSize) {display: flex;flex-flow: row wrap;margin: calc(gutterSize / -2);> div {flex: 0 0 calc(100% / columns);padding: calc(gutterSize / 2);box-sizing: border-box;} }2.用法 .grid-show-item3 {width: 100%;display: fl…...

测试微信模版消息推送

进入“开发接口管理”--“公众平台测试账号”&#xff0c;无需申请公众账号、可在测试账号中体验并测试微信公众平台所有高级接口。 获取access_token: 自定义模版消息&#xff1a; 关注测试号&#xff1a;扫二维码关注测试号。 发送模版消息&#xff1a; import requests da…...

C++初阶-list的底层

目录 1.std::list实现的所有代码 2.list的简单介绍 2.1实现list的类 2.2_list_iterator的实现 2.2.1_list_iterator实现的原因和好处 2.2.2_list_iterator实现 2.3_list_node的实现 2.3.1. 避免递归的模板依赖 2.3.2. 内存布局一致性 2.3.3. 类型安全的替代方案 2.3.…...

在HarmonyOS ArkTS ArkUI-X 5.0及以上版本中,手势开发全攻略:

在 HarmonyOS 应用开发中&#xff0c;手势交互是连接用户与设备的核心纽带。ArkTS 框架提供了丰富的手势处理能力&#xff0c;既支持点击、长按、拖拽等基础单一手势的精细控制&#xff0c;也能通过多种绑定策略解决父子组件的手势竞争问题。本文将结合官方开发文档&#xff0c…...

ffmpeg(四):滤镜命令

FFmpeg 的滤镜命令是用于音视频处理中的强大工具&#xff0c;可以完成剪裁、缩放、加水印、调色、合成、旋转、模糊、叠加字幕等复杂的操作。其核心语法格式一般如下&#xff1a; ffmpeg -i input.mp4 -vf "滤镜参数" output.mp4或者带音频滤镜&#xff1a; ffmpeg…...

Qt Http Server模块功能及架构

Qt Http Server 是 Qt 6.0 中引入的一个新模块&#xff0c;它提供了一个轻量级的 HTTP 服务器实现&#xff0c;主要用于构建基于 HTTP 的应用程序和服务。 功能介绍&#xff1a; 主要功能 HTTP服务器功能&#xff1a; 支持 HTTP/1.1 协议 简单的请求/响应处理模型 支持 GET…...

ServerTrust 并非唯一

NSURLAuthenticationMethodServerTrust 只是 authenticationMethod 的冰山一角 要理解 NSURLAuthenticationMethodServerTrust, 首先要明白它只是 authenticationMethod 的选项之一, 并非唯一 1 先厘清概念 点说明authenticationMethodURLAuthenticationChallenge.protectionS…...

select、poll、epoll 与 Reactor 模式

在高并发网络编程领域&#xff0c;高效处理大量连接和 I/O 事件是系统性能的关键。select、poll、epoll 作为 I/O 多路复用技术的代表&#xff0c;以及基于它们实现的 Reactor 模式&#xff0c;为开发者提供了强大的工具。本文将深入探讨这些技术的底层原理、优缺点。​ 一、I…...

Golang——6、指针和结构体

指针和结构体 1、指针1.1、指针地址和指针类型1.2、指针取值1.3、new和make 2、结构体2.1、type关键字的使用2.2、结构体的定义和初始化2.3、结构体方法和接收者2.4、给任意类型添加方法2.5、结构体的匿名字段2.6、嵌套结构体2.7、嵌套匿名结构体2.8、结构体的继承 3、结构体与…...

从 GreenPlum 到镜舟数据库:杭银消费金融湖仓一体转型实践

作者&#xff1a;吴岐诗&#xff0c;杭银消费金融大数据应用开发工程师 本文整理自杭银消费金融大数据应用开发工程师在StarRocks Summit Asia 2024的分享 引言&#xff1a;融合数据湖与数仓的创新之路 在数字金融时代&#xff0c;数据已成为金融机构的核心竞争力。杭银消费金…...

6️⃣Go 语言中的哈希、加密与序列化:通往区块链世界的钥匙

Go 语言中的哈希、加密与序列化:通往区块链世界的钥匙 一、前言:离区块链还有多远? 区块链听起来可能遥不可及,似乎是只有密码学专家和资深工程师才能涉足的领域。但事实上,构建一个区块链的核心并不复杂,尤其当你已经掌握了一门系统编程语言,比如 Go。 要真正理解区…...