linux并发服务器 —— 多线程并发(六)

线程概述

同一个程序中的所有线程均会独立执行相同程序,且共享同一份全局内存区域;

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

Linux环境下,线程的本质就是进程;

ps -Lf pid:查指定进程LWP号(线程号)

线程和进程的区别

1. 进程间的信息难以共享,除只读代码段,父子进程并未共享内存;

        线程共享信息方便快速(进程、父进程、进程组、会话ID,文件描述符表,当前工作目录,文件权限掩码,虚拟地址空间(除栈、.text));但超线程ID、信号掩码、error变量、调度策略和优先级、栈、本地变量不共享;

2. fork创建进程代价较高

        创建线程比创建进程快一个数量级以上

线程操作

/*#include <pthread.h>一般情况下,main所在线程为主线程/main线程,其余都成为子线程pthread_t pthread_self(void);功能:获取当前线程IDint pthread equal(pthread_t tl,pthread_t t2);功能:比较两个线程号是否相等不同操作系统,pthread_t类型实现不一样,有可能是结构体int pthread_create(pthread t *thread, const pthread attr t *attr,void *(*start_routine) (void *), void *arg);功能:创建一个子线程(调度的基本单位)参数:thread - 传出参数:线程创建成功,子线程ID会写入该变量attr - 设置线程的属性,默认值 - NULLstart_rountine - 函数指针,子线程需要处理的逻辑代码arg - 给start_rountine使用,传参返回值:成功 - 0失败 - 错误号,与errno不同;获取错误号信息:char* strerror(int errnum);void pthread_exit(void *retval);功能:终止一个当前调用线程参数:retval - 传递一个指针,作为一个返回值,可以在pthread_join中获取返回值: 没有任何返回值int pthread_join(pthread_t thread,void **retval);功能:和一个已经终止的线程进行连接回收子线程的资源这个函数是阻塞函数,调用一次只能回收一个子线程一般在主线程中去使用参数:thread - 需要回收的子线程IDretval - 接收子线程退出的返回值返回值:成功 - 0失败 - !0int pthread_detach(pthread_t thread);功能:分离一个线程,将线程标记分离,线程终止时自动释放资源给系统1. 不能多次分离,不可预料2. 不能去连接一个已经分离的线程,会报错(join)参数:需要分离的线程ID返回值:成功 - 0失败 - errorint pthread_cancel(pthread_t thread);功能:取消线程(让线程终止),中途暂停!但并不是立马终止,而是当一个子线程执行到一个取消点,线程才会终止取消点:系统规定好的一些系统调用,可以粗略认为是用户去到内核区的切换这个位置
*/

创建线程

#include <iostream>
#include <pthread.h>
#include <cstdio>
#include <string.h>
#include <unistd.h>
using namespace std;void* callback(void *arg){cout<<"子线程...."<<*((int*)arg)<<endl;return NULL;
}int main(){pthread_t tid;int num = 10;int ret = pthread_create(&tid , NULL , callback , (void*)&num);if(ret != 0){char* str = strerror(ret);cout<<"error: "<<str<<endl;}for(int i = 0 ; i<5 ; i++){cout<<i<<endl;}sleep(1);return 0;
}

终止线程

#include <iostream>
#include <pthread.h>
#include <cstdio>
#include <string.h>
#include <unistd.h>
using namespace std;void* callback(void *arg){char buf[1024];sprintf(buf , "子线程....%ld" , pthread_self());cout<<buf<<endl;return NULL;
}int main(){// 创建子线程pthread_t tid;int ret = pthread_create(&tid , NULL , callback , NULL);if(ret != 0){char* str = strerror(ret);cout<<str<<endl;}for(int i = 0 ; i<100 ; i++){cout<<i<<endl;}cout<<"子线程...."<<tid<<endl;cout<<"主线程...."<<pthread_self()<<endl;pthread_exit(NULL);// 主线程退出不会影响正常运行的线程return 0; // 进程退出 所有子线程立刻终止
}

链接已终止的线程

#include <iostream>
#include <pthread.h>
#include <cstdio>
#include <string.h>
#include <unistd.h>
using namespace std;int val = 10;void* callback(void *arg){char buf[1024];sprintf(buf , "子线程....%ld" , pthread_self());cout<<buf<<endl;// sleep(3);// return NULL; // pthread_exit(NULL);// int val = 10; // 局部变量pthread_exit((void*)&val);
}int main(){// 创建子线程pthread_t tid;int ret = pthread_create(&tid , NULL , callback , NULL);if(ret != 0){char* str = strerror(ret);cout<<str<<endl;}for(int i = 0 ; i<5 ; i++){cout<<i<<endl;}cout<<"子线程...."<<tid<<endl;cout<<"主线程...."<<pthread_self()<<endl;int* ptr;if(pthread_join(tid , (void **)&ptr) != 0){char* str = strerror(ret);cout<<str<<endl;}cout<<"回收子线程成功: "<<*(int *)ptr<<endl;pthread_exit(NULL);// 主线程退出不会影响正常运行的线程return 0; // 进程退出 所有子线程立刻终止
}

线程分离

#include <iostream>
#include <pthread.h>
#include <cstdio>
#include <string.h>
#include <unistd.h>
using namespace std;void* callback(void* arg){cout<<"我的ID: "<<pthread_self()<<endl;return NULL;
}int main(){// 创建pthread_t tid;int ret = pthread_create(&tid , NULL , callback , NULL);if(ret != 0){char* str = strerror(ret);cout<<"error1: "<<str<<endl;}cout<<"父线程:"<<pthread_self()<<"子线程:"<<tid<<endl;//子线程分离ret = pthread_detach(tid);if(ret != 0){char* str = strerror(ret);cout<<"error2: "<<str<<endl;}//对分离子线程进行连接ret = pthread_join(tid,NULL);if(ret != 0){char* str = strerror(ret);cout<<"error3: "<<str<<endl;}pthread_exit(NULL);return 0;
}

线程取消

#include <iostream>
#include <pthread.h>
#include <cstdio>
#include <string.h>
#include <unistd.h>
using namespace std;void* callback(void* arg){cout<<"我的ID: "<<pthread_self()<<endl;for(int i = 0 ; i<5 ; i++){cout<<"子线程:"<<i<<endl;}return NULL;
}int main(){// 创建pthread_t tid;int ret = pthread_create(&tid , NULL , callback , NULL);if(ret != 0){char* str = strerror(ret);cout<<"error1: "<<str<<endl;}// 取消线程pthread_cancel(tid);for(int i = 0 ; i<10 ; i++){cout<<i<<endl;}cout<<"父线程:"<<pthread_self()<<"子线程:"<<tid<<endl;pthread_exit(NULL);return 0;
}

线程属性

/*
int pthread_attr_init(pthread_attr_t *attr);初始化线程属性变量int pthread_attr_destroy(pthread_attr_t *attr);释放线程属性资源int pthread_attr_getdetachstate(const pthread_attq_t *attr, int* detachstate);获取线程分离的状态属性int pthread_attr_setdetachstate(pthread_attr_t *attr, int detachstate);设置线程分离的状态属性*/#include <iostream>
#include <pthread.h>
#include <cstdio>
#include <string.h>
#include <unistd.h>
using namespace std;void* callback(void* arg){cout<<"我的ID: "<<pthread_self()<<endl;return NULL;
}int main(){// 创建线程属性变量pthread_attr_t attr;pthread_attr_init(&attr);pthread_attr_setdetachstate(&attr , PTHREAD_CREATE_DETACHED);// 获取线程栈的大小size_t size;pthread_attr_getstacksize(&attr , &size);cout<<"子线程占空间大小:"<<size<<endl;// 创建pthread_t tid;int ret = pthread_create(&tid , &attr , callback , NULL);if(ret != 0){char* str = strerror(ret);cout<<"error1: "<<str<<endl;}cout<<"父线程:"<<pthread_self()<<"子线程:"<<tid<<endl;pthread_attr_destroy(&attr);ret = pthread_join(tid,NULL);if(ret != 0){char* str = strerror(ret);cout<<"error3: "<<str<<endl;}pthread_exit(NULL);return 0;
}

线程同步

必须确保多个线程不会同时修改同一变量,或者某一线程不会读取正在由其他线程修改的变量;

临界区是指访问某一共享资源的代码片段,这段代码的执行应该为原子操作(不能分割);

互斥锁

使用互斥锁来确保仅有一个线程可以访问某项共享资源,保证原子访问;

互斥锁由两种状态:锁定/未锁定,试图对锁定的互斥锁再加锁会导致线程阻塞/报错,取决于加锁使用的方法;

线程加锁成为互斥锁的所有者,只有所有者才能解锁;

/*
互斥量的类型 pthread_mutex_t
int pthread_mutex_init(pthread_mutex_t *restrict mutexconst pthread_mutexattr_t *restrict attr);功能:初始化互斥锁参数:mutex - 需要初始化的互斥锁attr - 互斥锁相关属性 NULLrestric - C语言修饰符,被修饰的指针不能由另外的指针进行操作
int pthread_mutex_destroy(pthread_mutex_t *mutex);释放互斥量的资源
int pthread_mutex_lock(pthread_mutex_t *mutex);加锁 , 如果有线程已经加锁,只能阻塞等待
int pthread_mutex_trylock(pthread_mutex_t *mutex);尝试加锁,加锁失败不会阻塞,会直接返回
int pthread_mutex_unlock(pthread_mutex_t *mutex);释放锁
*/

死锁

多个进程在执行过程中,因争夺共享资源而造成的一种互相等待的现象;

导致死锁的三个主要原因:

1. 加锁忘记释放

2. 重复枷锁

3. 线程之间对于锁循环等待

读写锁

读写锁允许多个读出,但只允许一个写入:

1. 如果有其他线程读数据,则允许其他线程执行读操作,但不允许写操作;

2. 有其他线程写数据,则其他线程不允许读/写;

3. 写是独占的,写的优先级高;

/*读写锁的类型 pthread_rwlock_tint pthread_rwlock_init(pthread_rwlock_t *restrict rwlock,const pthread_rwlockattr_t *restrict attr);int pthread_rwlock_destroy(pthread_rwlock_t *rwlock);int pthread_rwlock_rdlock(pthread_rwlock_t *rwlock);int pthread_rwlock_tryrdlock(pthread_rwlock_t *rwlock);int pthread_rwlock_wrlock(pthread_rwlock_t *rwlock);int pthread_rwlock_trywrlock(pthread_rwlock_t *rwlock);int pthread_rwlock_unlock(pthread_rwlock_t *rwlock);
*/
// 案例:创建8个线程,操作同一个全局变量;
//      3个线程不定时写一个全局变量,其余5个线程不定时读全局变量
#include <iostream>
#include <pthread.h>
#include <cstdio>
#include <unistd.h>
using namespace std;int num = 1;
// pthread_mutex_t mutex;
pthread_rwlock_t rwlock;void* wnum(void* arg){while(1){// pthread_mutex_lock(&mutex);pthread_rwlock_wrlock(&rwlock);num++;printf("++write , tid: %ld , num : %d\n" , pthread_self() , num);// pthread_mutex_unlock(&mutex);pthread_rwlock_unlock(&rwlock);sleep(1);}return NULL;
}void* rnum(void* arg){while(1){// pthread_mutex_lock(&mutex);pthread_rwlock_rdlock(&rwlock);printf("read , tid: %ld , num : %d\n" , pthread_self() , num);// pthread_mutex_unlock(&mutex);pthread_rwlock_unlock(&rwlock);sleep(1);}return NULL;
}int main(){// pthread_mutex_init(&mutex , NULL);pthread_rwlock_init(&rwlock , NULL);// 创建3个写线程 5个读线程pthread_t wtids[3] , rtids[5];for(int i = 0 ; i<3 ; i++){pthread_create(&wtids[i] , NULL , wnum , NULL);}for(int i = 0 ; i<5 ; i++){pthread_create(&rtids[i] , NULL , rnum , NULL);}// 设置线程分离for(int i = 0 ; i<3 ; i++){pthread_detach(wtids[i]);}for(int i = 0 ; i<5 ; i++){pthread_detach(rtids[i]);}pthread_exit(NULL);// pthread_mutex_destroy(&mutex);pthread_rwlock_destroy(&rwlock);return 0;
}

生产者消费者模式

多生产者 - 容器 - 多消费者

阻塞 - 通知机制,需要条件变量和信号量来进行实现;

条件变量 - 通过条件变量来唤醒阻塞进程

信号量 - 一定程度上表示资源的多少

/*信号量的类型 sem_tint sem_init(sem_t *sem, int pshared,unsigned int value);初始化信号量参数:sem - 信号量变量的地址pshared - 0用在线程,非0用在进程value - 信号量中的值int sem_destroy(sem_t *sem);释放资源int sem_wait(sem_t *sem);加锁 对信号量的值减1,如果值为0则阻塞int sem_trywait(sem_t *sem);尝试int sem_timedwait(sem_t *sem, const struct timespec *abs timeout);等待多少时间int sem_post(sem_t *sem);解锁 对信号量的值加1int sem_getvalue(sem_t *sem, int *sval);获取值
*/
// 生产者消费者模型 粗略版本
#include <iostream>
#include <pthread.h>
#include <cstdio>
#include <unistd.h>
#include <cstdlib>
#include <semaphore.h>
using namespace std;// 创建一个互斥锁
pthread_mutex_t mutex;
// 创建两个信号量
sem_t p , c;class Node{
public:int num;Node* next;
};Node* head = NULL;void* pro(void* arg){// 不断创建节点添加到链表while(1){sem_wait(&p);pthread_mutex_lock(&mutex);Node* newNode = new Node();newNode->next = head;head = newNode;newNode->num = rand()%100;printf("add node , num: %d , tid: %ld\n" , newNode->num , pthread_self());   pthread_mutex_unlock(&mutex);sem_post(&c);usleep(1000);}return NULL;
}void* cus(void* arg){while(1){sem_wait(&c);pthread_mutex_lock(&mutex);Node* cur = head;head = head->next;printf("del node : %d , tid : %ld\n" , cur->num , pthread_self());delete(cur);cur = NULL;pthread_mutex_unlock(&mutex);sem_post(&p);usleep(1000);}return NULL;
}int main(){pthread_mutex_init(&mutex , NULL);sem_init(&p , 0 , 5);sem_init(&c , 0 , 0);// 5个生产者,5个消费者pthread_t ptids[5] , ctids[5];for(int i = 0 ; i<5; i++){pthread_create(&ptids[i] , NULL , pro , NULL);pthread_create(&ctids[i] , NULL , cus , NULL);}for(int i = 0 ; i<5 ; i++){pthread_detach(ptids[i]);pthread_detach(ctids[i]);}while(1){sleep(10);}pthread_mutex_destroy(&mutex);pthread_exit(NULL);return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/121056.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

101序列检测器

本次所做设计&#xff0c;使用数字电路芯片实现的101序列检测器。电路图如下&#xff1a; 主要首先要根据需求画出状态转移方程&#xff0c;然后写出它的逻辑表达式。最后根据所选触发器种类确定电路图。序列由按键控制输入&#xff0c;按键按下&#xff0c;代表输入1 &#xf…

RT-Thread UART

UART 简介 UART&#xff08;Universal Asynchronous Receiver/Transmitter&#xff09;通用异步收发传输器&#xff0c;UART 作为异步串口通信协议的一种&#xff0c;工作原理是将传输数据的每个字符一位接一位地传输。是在应用程序开发过程中使用频率最高的数据总线。 UART …

Error: Cannot find module ‘timers/promises‘

这个错误很要命 他会导致你本机所有的npm 命令全部报错 首先 这个错误是因为 npm 与 node版本不匹配导致的 最简单的办法就是 查一下你安装的这个npm 的版本适配那个版本的 node 然后将本地的node删除 控制面板写在node 然后去官方文档现在与本地npm 匹配的node.js 这样 你执行…

【LeetCode-中等题】146. LRU 缓存

文章目录 题目方法一&#xff1a;直接继承LinkedHashMap调用api方法二&#xff1a;自定义LinkedHashMap HashMap ListNode LinkedHashMap 题目 LRU缓存是什么&#xff1a;LRU缓存机制&#xff0c;你想知道的这里都有 实现 LRU 缓存算法 方法一&#xff1a;直接继承Linked…

彻底学会Unity从网上加载资源到场景

使用类WWW 该类实例化的对象可以存储多种多媒体资源&#xff0c;只需要在构造函数中附上可访问的资源链接 Unity 中&#xff0c;WWW 类用于实例化互联网上的资源&#xff0c;如文本、图像、音频和视频等。WWW 实例化的对象可以存储多种多媒体素材。以下是一些常见的例子&…

固定资产卡片乱怎么管理

固定资产卡片是记录公司固定资产信息的重要工具&#xff0c;如果管理不善&#xff0c;容易造成卡片混乱、数据错误等问题。 为了避免这种情况的发生&#xff0c;可以采取以下措施&#xff1a;  建立完善的资产管理制度&#xff0c;明确固定资产的分类、标准和使用情况&#x…

C# 中什么是重写(子类改写父类方法)

方法重写是指在继承关系中&#xff0c;子类重新实现父类或基类的某个方法。这种方法允许子类根据需要修改或扩展父类或基类的方法功能。在面向对象编程中&#xff0c;方法重写是一种多态的表现形式&#xff0c;它使得子类可以根据不同的需求和场景提供不同的方法实现。 方法重…

热点探测技术架构设计与实践

1. 概述 说到热点问题&#xff0c;首先我们先理解一下什么是热点&#xff1f; 热点通常意义来说&#xff0c;是指在一段时间内&#xff0c;被广泛关注的物品或事件&#xff0c;例如微博热搜&#xff0c;热卖商品&#xff0c;热点新闻&#xff0c;明星直播等等&#xff0c;所以…

You must install at least one postgresql-client-<version> package

使用主机上的映射端口来连接到 PostgreSQL 数据库。例如&#xff0c;使用以下命令连接到数据库&#xff1a; psql -h localhost -p 5432 -U postgres出现下面的问题&#xff1a; 分析&#xff1a; 如果您在运行 psql 命令时遇到错误消息 You must install at least one pos…

应用案例 | 3D视觉引导解决方案汽车零部件上下料

Part.1 行业背景 三维视觉引导技术在国内外汽车零部件领域得到了广泛应用。随着汽车制造业的不断发展和创新&#xff0c;对于零部件的加工和装配要求越来越高&#xff0c;而三维视觉引导技术能够帮助企业实现更精确、更高效的零部件上下料过程。 纵览国外&#xff0c;部分汽车…

若依 MyBatis改为MyBatis-Plus

主要内容&#xff1a;升级成mybatis-plus&#xff0c;代码生成也是mybatis-plus版本 跟着我一步一步来&#xff0c;就可完成升级&#xff01; 检查&#xff1a;启动程序&#xff0c;先保证若依能启动 第一步&#xff1a;添加依赖 这里需要在两个地方添加&#xff0c;一个是最…

软件游戏丢失d3dcompiler_47.dll怎么办?这个几个解决方法可修复

当我们在玩软件游戏时&#xff0c;有时候会出现丢失 d3dcompiler_47.dll 的问题&#xff0c;这让我们感到非常困扰。d3dcompiler_47.dll 是 DirectX 中的一个重要组件&#xff0c;如果它丢失了&#xff0c;那么很多游戏就无法正常运行。我将和大家分享一下我在解决软件游戏丢失…

计算机竞赛 基于深度学习的中文情感分类 - 卷积神经网络 情感分类 情感分析 情感识别 评论情感分类

文章目录 1 前言2 情感文本分类2.1 参考论文2.2 输入层2.3 第一层卷积层&#xff1a;2.4 池化层&#xff1a;2.5 全连接softmax层&#xff1a;2.6 训练方案 3 实现3.1 sentence部分3.2 filters部分3.3 featuremaps部分3.4 1max部分3.5 concat1max部分3.6 关键代码 4 实现效果4.…

小白学go基础03-了解Go项目的项目结构

我们先来看看第一个Go项目——Go语言自身——的项目结构是什么样的。Go项目的项目结构自1.0版本发布以来一直十分稳定&#xff0c;直到现在Go项目的顶层结构基本没有大的改变。 截至Go项目commit 1e3ffb0c&#xff08;2019.5.14&#xff09;&#xff0c;Go1.0 项目结构如下&am…

Ansible学习笔记15

1、roles&#xff1a;&#xff08;难点&#xff09; roles介绍&#xff1a; roles&#xff08;角色&#xff09;&#xff1a;就是通过分别将variables&#xff0c;tasks及handlers等放置于单独的目录中&#xff0c;并可以便捷地调用他们的一种机制。 假设我们要写一个playbo…

KaTex用法

KaTeX是一个用于数学公式渲染的JavaScript库&#xff0c;可以在网页上方便地显示数学符号和公式。下面是KaTeX的使用方法&#xff1a; 在网页中引入KaTeX的CSS和JS文件&#xff1a; <link rel"stylesheet" href"https://cdnjs.cloudflare.com/ajax/libs/Ka…

【真题解析】系统集成项目管理工程师 2023 年上半年真题卷(综合知识)

本文为系统集成项目管理工程师考试(软考) 2023 年上半年真题(全国卷),包含答案与详细解析。考试共分为两科,成绩均 ≥45 即可通过考试: 综合知识(选择题 75 道,75分)案例分析(问答题 4 道,75分)综合知识(选择题*75)1-10 题11-20 题21-30 题31-40 题41-50 题51-60 …

webpack(三)loader

定义 loader用于对模块的源代码进行转换&#xff0c;在imporrt或加载模块时预处理文件 webpack做的事情&#xff0c;仅仅是分析出各种模块的依赖关系&#xff0c;然后形成资源列表&#xff0c;最终打包生成到指定文件中。 在webpack内部&#xff0c;任何文件都是模块&#x…

VBA技术资料MF52:VBA_在Excel中突出显示前 10 个值

【分享成果&#xff0c;随喜正能量】一言之善&#xff0c;重于千金。善良不分大小&#xff0c;有时候你以为的一句话&#xff0c;小小的举手之劳&#xff0c;也可能就是别人的救赎&#xff01;不要吝啬你的善良&#xff0c;因为你永远不知道那小小的善良能给多少人带来光明。。…

大厂面试 | 百度一面,顶不住

题目来源&#xff1a;https://www.nowcoder.com/feed/main/detail/d39aabc0debd4dba810b4b9671d54348 前文 本期是【捞捞面经】系列文章的第 2 期&#xff0c;持续更新中…。&#xff08;更多与往期下方仓库直达&#xff09; 《捞捞面经》系列正式开始连载啦&#xff0c;据说看…