当前位置:   article > 正文

三、【C++项目】WebServer-多线程开发-详细笔记_webserver c++项目

webserver c++项目

目录 

 3.1线程

01/线程概述

与进程process类似,线程thread是允许应用程序并发执行多个任务的一种机制。一个进程可以包含多个线程。

进程是CPU分配资源的最小单位,线程是操作系统调度执行的最小单位。

线程之间共享和非共享的资源

 线程操作

  1. int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine) (void *), void *arg); //创建线程
  2. pthread_t pthread_self(void); //获取线程号ID
  3. int pthread_equal(pthread_t t1, pthread_t t2);//比较两个线程是否相等
  4. void pthread_exit(void *retval); //终止一个线程,在哪个线程中调用,就表示终止哪个线程
  5. int pthread_join(pthread_t thread, void **retval); //和一个已经终止的线程进行连接(回收资源)
  6. int pthread_detach(pthread_t thread); //分离一个线程;被分离的线程在终止的时候,会自动释放资源返回给系统。
  7. int pthread_cancel(pthread_t thread); //取消线程(让线程终止)

3.2创建线程

一般情况,main函数所在的线程我们称为主线程(main线程),其余创建的线程成为子线程。

程序中默认只有一个进程,fork()函数调用,2进程

程序中默认只有一个线程,pthread_create()函数调用,2个线程

  1. #include <pthread.h>
  2.  int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine) (void *), void *arg);
  3. /*
  4.         - 功能:创建一个子线程
  5.         - 参数:
  6.             - thread:传出参数,线程创建成功后,子线程的线程ID被写到该变量中。
  7.             - attr : 设置线程的属性,一般使用默认值,NULL
  8.             - start_routine : 函数指针,这个函数是子线程需要处理的逻辑代码
  9.             - arg : 给第三个参数使用,传参
  10.         - 返回值:
  11.             成功:0
  12.             失败:返回错误号。这个错误号和之前errno不太一样。
  13.             获取错误号的信息:  char * strerror(int errnum);
  14. */

代码

  1. #include <stdio.h>
  2. #include <pthread.h>
  3. #include <string.h>
  4. #include <unistd.h>
  5. void * callback(void * arg) {
  6.     printf("child thread...\n");
  7.     printf("arg value: %d\n", *(int *)arg);
  8.     return NULL;
  9. }
  10. int main() {
  11.     pthread_t tid;
  12.     int num = 10;
  13.     // 创建一个子线程
  14.     int ret = pthread_create(&tid, NULL, callback, (void *)&num);
  15.     if(ret != 0) {
  16.         char * errstr = strerror(ret);
  17.         printf("error : %s\n", errstr);
  18.     }
  19.     for(int i = 0; i < 5; i++) {
  20.         printf("%d\n", i);
  21.     }
  22.     sleep(1);
  23.     return 0;   // exit(0);
  24. }

执行 gcc pthread_create.c -o create

报错,

 原因:这个线程pthread_create是第三方的库,不是标准库,需要指定库的名称,用 -l

gcc pthread_create.c -o create -l pthread

3.3终止线程

  1. #include <pthread.h>
  2. void pthread_exit(void *retval);
  3. /*
  4.      功能:终止一个线程,在哪个线程中调用,就表示终止哪个线程
  5.      参数:
  6.          retval:需要传递一个指针,作为一个返回值,可以在pthread_join()中获取到。
  7. */
  8. pthread_t pthread_self(void);
  9.      //功能:获取当前的线程的线程ID
  10. int pthread_equal(pthread_t t1, pthread_t t2);
  11.     //功能:比较两个线程ID是否相等
  12.     //不同的操作系统,pthread_t类型的实现不一样,有的是无符号的长整型,有的是使用结构体去实现的。
  13.    

代码

  1. #include <stdio.h>
  2. #include <pthread.h>
  3. #include <string.h>
  4. void * callback(void * arg) {
  5.     printf("child thread id : %ld\n", pthread_self());
  6.     return NULL;    // pthread_exit(NULL);
  7. }
  8. int main() {
  9.     // 创建一个子线程
  10.     pthread_t tid;
  11.     int ret = pthread_create(&tid, NULL, callback, NULL);
  12.     if(ret != 0) {
  13.         char * errstr = strerror(ret);
  14.         printf("error : %s\n", errstr);
  15.     }
  16.     // 主线程
  17.     for(int i = 0; i < 5; i++) {
  18.         printf("%d\n", i);
  19.     }
  20.     printf("tid : %ld, main thread id : %ld\n", tid ,pthread_self());
  21.     // 让主线程退出,当主线程退出时,不会影响其他正常运行的线程。
  22.     pthread_exit(NULL);
  23.     printf("main thread exit\n");
  24.     return 0;   // exit(0);
  25. }

3.4 连接已终止多线程 

  1. #include <pthread.h>
  2. int pthread_join(pthread_t thread, void **retval);
  3. /*
  4.         - 功能:和一个已经终止的线程进行连接
  5.                 回收子线程的资源
  6.                 这个函数是阻塞函数,调用一次只能回收一个子线程
  7.                 一般在主线程中使用
  8.         - 参数:
  9.             - thread:需要回收的子线程的ID
  10.             - retval: 接收子线程退出时的返回值
  11.         - 返回值:
  12.             0 : 成功
  13.             非0 : 失败,返回的错误号
  14. */

代码 

  1. #include <stdio.h>
  2. #include <pthread.h>
  3. #include <string.h>
  4. #include <unistd.h>
  5. int value = 10; //全局变量
  6. void * callback(void * arg) {
  7.     printf("child thread id : %ld\n", pthread_self());
  8.     // sleep(3);
  9.     // return NULL;
  10.  
  11.     // int value = 10; // 局部变量
  12.     pthread_exit((void *)&value);   // return (void *)&value;
  13. }
  14. int main() {
  15.     // 创建一个子线程
  16.     pthread_t tid;
  17.     int ret = pthread_create(&tid, NULL, callback, NULL);
  18.     if(ret != 0) {
  19.         char * errstr = strerror(ret);
  20.         printf("error : %s\n", errstr);
  21.     }
  22.     // 主线程
  23.     for(int i = 0; i < 5; i++) {
  24.         printf("%d\n", i);
  25.     }
  26.     printf("tid : %ld, main thread id : %ld\n", tid ,pthread_self());
  27.     // 主线程调用pthread_join()回收子线程的资源
  28.     int * thread_retval;
  29.     ret = pthread_join(tid, (void **)&thread_retval);
  30.     if(ret != 0) {
  31.         char * errstr = strerror(ret);
  32.         printf("error : %s\n", errstr);
  33.     }
  34.     printf("exit data : %d\n", *thread_retval);
  35.     printf("回收子线程资源成功!\n");
  36.     // 让主线程退出,当主线程退出时,不会影响其他正常运行的线程。
  37.     pthread_exit(NULL);
  38.     return 0;
  39. }

3.5 线程的分离

  1. #include <pthread.h>
  2. int pthread_detach(pthread_t thread);
  3. /*
  4.         - 功能:分离一个线程。被分离的线程在终止的时候,会自动释放资源返回给系统。
  5.           1.不能多次分离,会产生不可预料的行为。
  6.           2.不能去连接一个已经分离的线程,会报错。
  7.         - 参数:需要分离的线程的ID
  8.         - 返回值:
  9.             成功:0
  10.             失败:返回错误号
  11. */

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/木道寻08/article/detail/889280
推荐阅读
相关标签
  

闽ICP备14008679号