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

C++11 多线程编程-小白零基础到手撕线程池

提示:文章

文章目录

  • 前言
  • 一、背景
  • 二、
    • 2.1
    • 2.2
  • 总结

前言

前期疑问:
本文目标:


一、背景

来源于b站视频

C++11 多线程编程-小白零基础到手撕线程池

学习来源:https://www.bilibili.com/video/BV1d841117SH/?p=2&spm_id_from=pageDriver&vd_source=1a54eaaaa0e36b9ef70e2dbe59d5b137

http://www.seestudy.cn/?list_9/35.html

相关知识点

thread创建线程
join
detach
joinable
std::ref
智能指针在头文件中

二 、互斥量

自己写的代码

2.1 互斥量

#include <iostream>
#include <thread>int counter = 0;void fun()
{for (int i = 0; i < 10000; i++){counter++;}
}int main() {std::thread t1(fun);std::thread t2(fun);t1.join();t2.join();std::cout << "a:" << counter << std::endl;
}

没有实现竞争访问a导致结果不是2000的情况,我这边的打印结果是20000。

看有的评论说加一个0实现了,我加了好几个0还是没有实现。

有的评论说现在编译器都是2000了,不明所以,继续向下看吧。


第二天来用了课程对应的代码重新跑了一下,同时循环次数尝试加了个0。又出现了变量打印的值不是20000的情况。而我的myFirstTest工程依然出现不了预期的情况,先不管了。

2.2 互斥量锁死

#include <iostream>
#include <thread>
#include <mutex>std::mutex m1, m2;int counter = 0;//死锁
void fun1()
{for (int i = 0; i < 5000; i++){m1.lock();m2.lock();m1.unlock();m2.unlock();}
}void fun2()
{for (int i = 0; i < 5000; i++){m2.lock();m1.lock();m1.unlock();m2.unlock();}
}void fun()
{for (int i = 0; i < 1000000; i++){counter++;}
}int main() {std::thread s1(fun1);std::thread s2(fun2);s1.join();s2.join();std::thread t1(fun);std::thread t2(fun);t1.join();t2.join();std::cout << "a:" << counter << std::endl;
}

循环次数50、500都不会锁死,5000会锁死。

视频中给出的防止死锁的解决办法是,每个线程都先获取m1,继续获取m2。顺序获取。可以防止死锁。

(2024年9月29日17:11:45 今天再看代码,理解之前写的【顺序获取。可以防止死锁。】,应该是修改上述代码为都先获取m1,再获取m2)

5、std::lock_guard

#include <iostream>
#include <thread>
#include <mutex>std::mutex tex;
std::mutex m1, m2;int counter = 0;void fun()
{for (int i = 0; i < 1000000; i++){std::lock_guard<std::mutex> lg(tex);counter++;}
}int main() {std::thread t1(fun);std::thread t2(fun);t1.join();t2.join();std::cout << "a:" << counter << std::endl;
}

lock_guard源码

template<class _Mutex>class lock_guard{	// class with destructor that unlocks a mutex
public:using mutex_type = _Mutex;explicit lock_guard(_Mutex& _Mtx): _MyMutex(_Mtx){	// construct and lock_MyMutex.lock();}lock_guard(_Mutex& _Mtx, adopt_lock_t): _MyMutex(_Mtx){	// construct but don't lock}~lock_guard() noexcept{	// unlock_MyMutex.unlock();}lock_guard(const lock_guard&) = delete;lock_guard& operator=(const lock_guard&) = delete;
private:_Mutex& _MyMutex;};

针对这个源码,其中_MyMutex是私有成员变量。explicit表示禁止隐式转换。还涉及到构造函数重载、禁用构造、禁用拷贝函数。

2024年9月29日17:21:30

这篇文章是在之前写的,后面我又因为看代码疑惑信号量的使用,又写了下面的一篇文章:关于多线程unique_lock和guard_lock,而实际我在写关于多线程unique_lock和guard_lock这篇文章的时候也没有想起这边写的关于信号量的知识点。然后这次在看到这篇文章就想到了后面写的文章关于多线程unique_lock和guard_lock。但是我也忘了关于多线程unique_lock和guard_lock这篇文章的内容了。所以两篇文章结合看了下,加深了理解。

… …

七、 std::call_once与其使用场景

涉及到单例类。两种模式

饿汉模式和懒汉模式。

教程中以log类举例子,

static Log& GetInstance()
{static Log log;return log;
}
//这种是饿汉模式,构建类的时候就创建Log静态类对象。
//然后我的疑问是每次GetInstance的时候,不会多次创建log对象吗?实际是log是静态成员,只有一个

下面的是懒汉对象

static Log& GetInstance()
{static Log *log = nullptr;if(!log) {log = new Log();}return log;
}

上面代码我的疑问是,每次GetInstance的时候不会多次new对象吗?实际是不会,因为if(!log)做了判断,log不为空就不会再new对象。

针对饿汉模式在构造函数中创建静态变量,这边为什么不会继续创建对象,我查了资料,没看到啥解释。

但是看到另外两个点。

第一个就是懒汉模式申请的堆内存如果释放会内存泄漏。我觉得可以在析构函数释放就可以。

第二个就是帖子提到单例类模式线程不安全,主要是懒汉模式,多个线程读取if(!log)中log变量的时候,可能会多次申请堆内存。可以在if(!log)增加互斥锁,但是会影响效率。

参考文档:https://blog.csdn.net/code_feien/article/details/110423021

针对上述静态变量的问题,我还写了下面的测试代码

#include <iostream>using namespace std;int getData()
{static int test = 0;test++;return test;
}int main() {std::cout << "Hello, World!" << std::endl;for(int i = 0; i < 10; i++){int num = getData();cout << num << endl;}return 0;
}//预测结果是10
//实际打印结果1——10,确实和预期一样
//码可以理解为静态变量已经创建生命周期一直存在直到程序结束

九、 线程池

threadPool.h文件

#include <iostream>
#include <thread>
#include <mutex>
#include <condition_variable>
#include <functional>
#include <queue>class ThreadPool {
public:ThreadPool(int numThreads) : stop(false) {for (int i = 0; i < numThreads; ++i) {threads.emplace_back([this] {while (true) {std::unique_lock<std::mutex> lock(mutex);condition.wait(lock, [this] { return stop || !tasks.empty(); });if (stop && tasks.empty()) {return;}std::function<void()> task(std::move(tasks.front()));tasks.pop();lock.unlock();task();}});}}~ThreadPool() {{std::unique_lock<std::mutex> lock(mutex);stop = true;}condition.notify_all();for (std::thread& thread : threads) {thread.join();}}template<typename F, typename... Args>void enqueue(F&& f, Args&&... args) {std::function<void()> task(std::bind(std::forward<F>(f), std::forward<Args>(args)...));{std::unique_lock<std::mutex> lock(mutex);tasks.emplace(std::move(task));}condition.notify_one();}private:std::vector<std::thread> threads;std::queue<std::function<void()>> tasks;std::mutex mutex;std::condition_variable condition;bool stop;
};

main.cpp

#include "threadPool.h"int main(void)
{ThreadPool pool(4);for (int i = 0; i < 8; ++i) {pool.enqueue([i] {std::cout << "Task " << i << " is running in thread " << std::this_thread::get_id() << std::endl;std::this_thread::sleep_for(std::chrono::seconds(1));std::cout << "Task " << i << " is done" << std::endl;});}return 0;
}

关于上述代码设计到lamda等知识点

9.1 std:bind

其中关于std::bind,见这篇文章: std::bind的讲解

9.2 std::forward

然后还有一个知识点就是std::forward,

参考这篇文章:std::forward与完美转发详解

然后我继续查找关于std::forward知识点,看到这个例子

std::forward入门

std::forward 是 C++11 引入的标准库函数,用于实现完美转发。完美转发意味着在函数模板内,保持传递给函数的参数的左右值属性和常量属性。

下面是一个简单的 std::forward 使用示例:

#include <iostream>
#include <utility>// 这是一个函数模板,用来展示完美转发的效果
template<typename T>
void printValue(T&& val) {// 使用std::forward保留val的左右值属性和常量属性std::cout << (std::is_same<T, int&>::value ? "LValue: " : "RValue: ")<< std::forward<T>(val) << std::endl;
}int main() {int a = 5;printValue(a); // 将a作为左值传递printValue(std::move(a)); // 将a转换为右值,并传递return 0;
}

在这个例子中,printValue 是一个函数模板,它接受一个模板参数可以是任何类型的右值引用。当我们调用 printValue 时,我们可以传递一个左值或者右值。std::forward 保证在函数内部,我们传入的参数在模板实例化时保持其原有的左右值属性和常量属性。

输出结果将会是:

LValue: 5
RValue: 5

上述我不知道为什么上面的例子就能表现出std::forward的完美转发,我目前理解为std::forward可以实现左值和右值的转发。

然后上面有涉及到一个点就是std::is_same,关于std::is_same参考这篇文章:C++ 语言 std::is_same

9.3 std::function

std::function详解

我在纠结**std::function<void()> task(std::move(tasks.front()));**这个std::function<void()>是什么意思。查到下面的一个文章

c++ std::function的使用

其中一个示例

#include <functional>void function1()
{std::cout << "This is function1." << std::endl;
}int main()
{// 使用函数指针初始化 std::function 对象std::function<void()> f1 = function1;f1();
}//打印结果
//This is function1.

我理解大概意思就是std::function<void()>就是可以接收一个void f()函数。

9.4 对这个线程池的理解

lamda写法不是很理解,一直想找一个c++写的线程池代码没找到。

看了b站陈子青的视频,使用了他写的线程池,即有很多lamda表达式的线程池代码。可以实现900多个文件的正常读写。


总结

未完待续

相关文章:

C++11 多线程编程-小白零基础到手撕线程池

提示&#xff1a;文章 文章目录 前言一、背景二、 2.1 2.2 总结 前言 前期疑问&#xff1a; 本文目标&#xff1a; 一、背景 来源于b站视频 C11 多线程编程-小白零基础到手撕线程池 学习来源&#xff1a;https://www.bilibili.com/video/BV1d841117SH/?p2&spm_id_f…...

智源研究院与百度达成战略合作 共建AI产研协同生态

2024年9月24日&#xff0c;北京智源人工智能研究院&#xff08;简称“智源研究院”&#xff09;与北京百度网讯科技有限公司&#xff08;简称“百度”&#xff09;正式签署战略合作协议&#xff0c;双方将充分发挥互补优势&#xff0c;在大模型等领域展开深度合作&#xff0c;共…...

Flask-SQLAlchemy:在Flask应用中优雅地操作数据库

在Python的Web开发领域&#xff0c;Flask是一个备受欢迎的轻量级Web框架&#xff0c;它以简洁、灵活而著称。而当我们需要在Flask应用中与数据库进行交互时&#xff0c;Flask-SQLAlchemy就成为了一个强大而便捷的工具。它将Flask的简洁性与SQLAlchemy的强大数据库抽象能力完美结…...

智能巡检机器人 数据库

智能巡检机器人AI智能识别。无需人工。只需后台监控结果即可&#xff01;...

Spring AOP异步操作实现

在Spring框架中&#xff0c;AOP&#xff08;面向切面编程&#xff09;提供了一种非常灵活的方式来增强应用程序的功能。异步操作是现代应用程序中常见的需求&#xff0c;尤其是在处理耗时任务时&#xff0c;它可以帮助我们提高应用程序的响应性和吞吐量。Spring提供了一种简单的…...

【2006.07】UMLS工具——MetaMap原理深度解析

文献&#xff1a;《MetaMap: Mapping Text to the UMLS Metathesaurus》2006 年 7 月 14 日 https://lhncbc.nlm.nih.gov/ii/information/Papers/metamap06.pdf MetaMap&#xff1a;将文本映射到 UMLS 元数据库 总结 解决的问题 自动概念映射问题&#xff1a;解决如何将文本…...

ros2 colcon build 构建后,install中的local_setup.bash 和setup.bash有什么区别

功能概述 在 ROS2 中&#xff0c;colcon build是用于构建软件包的工具。构建完成后会生成install文件夹&#xff0c;其中的setup.bash和local_setup.bash文件都与环境设置相关&#xff0c;但存在一些区别。setup.bash 作用范围 setup.bash文件用于设置整个工作空间的环境变量。…...

Thymeleaf基础语法

Thymeleaf 是一种用于 Web 和非 Web 环境的现代服务器端 Java 模板引擎。它能够处理 HTML、XML、JavaScript、CSS 甚至纯文本。以下是 Thymeleaf 的一些基础语法&#xff1a; 1. 变量表达式 <!-- 显示变量的值 --> <p th:text"${name}">Default Name&l…...

spring cloud alibaba学习路线

以下是一条学习Spring Cloud Alibaba的路线&#xff1a; 一、基础前置知识 1. Java基础 熟练掌握Java语言特性&#xff0c;包括面向对象编程、集合框架、多线程等知识。 2. Spring和Spring Boot基础深入理解Spring框架&#xff0c;如依赖注入&#xff08;DI&#xff09;、控…...

基于 Seq2Seq 的中英文翻译项目(pytorch)

项目简介 本项目旨在使用 PyTorch 构建一个基于 Seq2Seq(编码器-解码器架构)的中英文翻译模型。我们将使用双语句子对的数据进行训练,最终实现一个能够将英文句子翻译为中文的模型。项目的主要步骤包括: 数据预处理:从数据集中提取英文和中文句子,并进行初步清洗和保存。…...

部标主动安全(ADAS+DMS)对接说明

1.前言 上一篇介绍了部标&#xff08;JT/T1078&#xff09;流媒体对接说明&#xff0c;这里说一下如何对接主动安全附件服务器。 流媒体的对接主要牵扯到4个方面&#xff1a; &#xff08;1&#xff09;平台端&#xff1a;业务端系统&#xff0c;包含前端呈现界面。 &#x…...

C++ STL(1)迭代器

文章目录 一、迭代器详解1、迭代器的定义与功能2、迭代器类型3、示例4、迭代器失效4.1、vector 迭代器失效分析4.2、list 迭代器失效分析4.3、set 与 map 迭代器失效分析 5、总结 前言&#xff1a; 在C标准模板库&#xff08;STL&#xff09;中&#xff0c;迭代器是一个核心概念…...

uview表单校验不生效问题

最近几次使用发现有时候会不生效&#xff0c;具体还没排查出来什么原因&#xff0c;先记录一下解决使用方法 <u--formlabelPosition"top"labelWidth"auto":model"form":rules"rules"ref"uForm" ><view class"…...

前端开发设计模式——单例模式

目录 一、单例模式的定义和特点&#xff1a; 1.定义&#xff1a; 2.特点&#xff1a; 二、单例模式的实现方式&#xff1a; 1.立即执行函数结合闭包实现&#xff1a; 2.ES6类实现&#xff1a; 三、单例模式的应用场景 1.全局状态管理&#xff1a; 2.日志记录器&#xff1a; …...

行情叠加量化,占据市场先机!

A股久违的3000点&#xff0c;最近都没有更新&#xff0c;现在终于对我们的市场又来点信息。相信在座的朋友这几天都是喜笑颜开&#xff0c;对A股又充满信心。当前行情好起来了&#xff0c;很多朋友又开始重回市场&#xff0c;研究股票学习量化&#xff0c;今天我们给大家重温下…...

大厂面试真题-ConcurrentHashMap怎么保证的线程安全?

ConcurrentHashMap是Java中的一个线程安全的哈希表实现&#xff0c;它通过一系列精妙的机制来保证线程安全。以下是ConcurrentHashMap保证线程安全的主要方式&#xff1a; 分段锁&#xff08;Segment Locking&#xff0c;Java 1.8之前&#xff09;&#xff1a; 在Java 1.8之前的…...

【RabbitMQ】消息堆积、推拉模式

消息堆积 原因 消息堆积是指在消息队列中&#xff0c;待处理的消息数量超过了消费者处理能力&#xff0c;导致消息在队列中不断堆积的现象。通常有以下几种原因&#xff1a; 消息生产过快&#xff1a;在高流量或者高负载的情况下&#xff0c;生产者以极高的速率发送消息&…...

MySQL常用SQL语句(持续更新中)

文章目录 数据库相关表相关索引相关添加索引 编码相关系统变量相关 收录一些经常用到的sql 数据库相关 建数据库 CREATE DATABASE [IF NOT EXISTS] <数据库名> [[DEFAULT] CHARACTER SET <字符集名>] [[DEFAULT] COLLATE <校对规则名>];例如&#xff1a; C…...

【更新】红色文化之红色博物馆数据集(经纬度+地址)

数据简介&#xff1a;红色博物馆作为国家红色文化传承与爱国主义教育的重要基地&#xff0c;遍布全国各地&#xff0c;承载着丰富的革命历史与文化记忆。本数据说明旨在汇总并分析全国范围内具有代表性的红色博物馆的基本信息&#xff0c;包括其地址、特色及教育意义&#xff0…...

Python项目Flask框架整合Redis

一、在配置文件中创建Redis连接信息 二、 实现Redis配置类 import redis from config.config import REDIS_HOST, REDIS_PORT, REDIS_PASSWD, REDIS_DB, EXPIRE_TIMEclass RedisDb():def __init__(self, REDIS_HOST, REDIS_PORT, REDIS_DB, EXPIRE_TIME, REDIS_PASSWD):# 建立…...

React Native 开发环境搭建(全平台详解)

React Native 开发环境搭建&#xff08;全平台详解&#xff09; 在开始使用 React Native 开发移动应用之前&#xff0c;正确设置开发环境是至关重要的一步。本文将为你提供一份全面的指南&#xff0c;涵盖 macOS 和 Windows 平台的配置步骤&#xff0c;如何在 Android 和 iOS…...

LeetCode - 394. 字符串解码

题目 394. 字符串解码 - 力扣&#xff08;LeetCode&#xff09; 思路 使用两个栈&#xff1a;一个存储重复次数&#xff0c;一个存储字符串 遍历输入字符串&#xff1a; 数字处理&#xff1a;遇到数字时&#xff0c;累积计算重复次数左括号处理&#xff1a;保存当前状态&a…...

【机器视觉】单目测距——运动结构恢复

ps&#xff1a;图是随便找的&#xff0c;为了凑个封面 前言 在前面对光流法进行进一步改进&#xff0c;希望将2D光流推广至3D场景流时&#xff0c;发现2D转3D过程中存在尺度歧义问题&#xff0c;需要补全摄像头拍摄图像中缺失的深度信息&#xff0c;否则解空间不收敛&#xf…...

工程地质软件市场:发展现状、趋势与策略建议

一、引言 在工程建设领域&#xff0c;准确把握地质条件是确保项目顺利推进和安全运营的关键。工程地质软件作为处理、分析、模拟和展示工程地质数据的重要工具&#xff0c;正发挥着日益重要的作用。它凭借强大的数据处理能力、三维建模功能、空间分析工具和可视化展示手段&…...

vue3 定时器-定义全局方法 vue+ts

1.创建ts文件 路径&#xff1a;src/utils/timer.ts 完整代码&#xff1a; import { onUnmounted } from vuetype TimerCallback (...args: any[]) > voidexport function useGlobalTimer() {const timers: Map<number, NodeJS.Timeout> new Map()// 创建定时器con…...

第 86 场周赛:矩阵中的幻方、钥匙和房间、将数组拆分成斐波那契序列、猜猜这个单词

Q1、[中等] 矩阵中的幻方 1、题目描述 3 x 3 的幻方是一个填充有 从 1 到 9 的不同数字的 3 x 3 矩阵&#xff0c;其中每行&#xff0c;每列以及两条对角线上的各数之和都相等。 给定一个由整数组成的row x col 的 grid&#xff0c;其中有多少个 3 3 的 “幻方” 子矩阵&am…...

如何在网页里填写 PDF 表格?

有时候&#xff0c;你可能希望用户能在你的网站上填写 PDF 表单。然而&#xff0c;这件事并不简单&#xff0c;因为 PDF 并不是一种原生的网页格式。虽然浏览器可以显示 PDF 文件&#xff0c;但原生并不支持编辑或填写它们。更糟的是&#xff0c;如果你想收集表单数据&#xff…...

Angular微前端架构:Module Federation + ngx-build-plus (Webpack)

以下是一个完整的 Angular 微前端示例&#xff0c;其中使用的是 Module Federation 和 npx-build-plus 实现了主应用&#xff08;Shell&#xff09;与子应用&#xff08;Remote&#xff09;的集成。 &#x1f6e0;️ 项目结构 angular-mf/ ├── shell-app/ # 主应用&…...

sipsak:SIP瑞士军刀!全参数详细教程!Kali Linux教程!

简介 sipsak 是一个面向会话初始协议 (SIP) 应用程序开发人员和管理员的小型命令行工具。它可以用于对 SIP 应用程序和设备进行一些简单的测试。 sipsak 是一款 SIP 压力和诊断实用程序。它通过 sip-uri 向服务器发送 SIP 请求&#xff0c;并检查收到的响应。它以以下模式之一…...

Python Ovito统计金刚石结构数量

大家好,我是小马老师。 本文介绍python ovito方法统计金刚石结构的方法。 Ovito Identify diamond structure命令可以识别和统计金刚石结构,但是无法直接输出结构的变化情况。 本文使用python调用ovito包的方法,可以持续统计各步的金刚石结构,具体代码如下: from ovito…...