webserver项目

利用无锁工作队列的Web服务器设计

项目地址https://github.com/whitehat32/webserver_no_lock
基本流程与牛客版的一致,下面放一个牛客版的流程框图
在这里插入图片描述

引言

在Web服务器的设计与实现中,性能优化是永远不会过时的话题。一般来说,Web服务器需要能够有效地处理大量并发请求。在多线程环境中,工作队列的设计尤为关键。传统的工作队列通常涉及使用锁(例如互斥锁)来确保线程安全,但这可能导致性能瓶颈。本博客文章将探讨一种全新的Web服务器设计,其主要特点是工作队列在访问任务时不使用锁。

为什么避免使用锁?

在多线程环境下,许多Web服务器使用锁来确保多线程能够安全地访问共享资源。但是,锁操作可能导致线程阻塞,进而增加CPU上下文切换,影响性能。

/* 线程池部分的代码 */
#ifndef THREADPOOL_H
#define THREADPOOL_H
// ...(省略部分代码)
std::thread([pool = pool_, p=i] {while(true) {if(!pool->tasks[p].empty()) {std::function<void()> task;if(pool->tasks[p].pop(task)) task();else continue;} else if(pool->isClosed) break;}
}).detach();
// ...(省略部分代码)
#endif //THREADPOOL_H

上面的代码片段展示了如何在多线程环境中避免使用锁。这是通过使用无锁队列(Lock-Free Queue)实现的,该队列使用原子操作来确保线程安全。

无锁工作队列的设计与实现

数据结构选择

本博文选择了单生产者单消费者(SPSC)无锁队列作为基础数据结构。这样可以利用原子操作来避免传统锁带来的性能问题。

// 无锁队列定义
/** File:   SpScLockFreeQueue.h* Author: Sander Jobing** Created on July 29, 2017, 5:17 PM** This class implements a Single Producer - Single Consumer lock-free and* wait-free queue. Only 1 thread can fill the queue, another thread can read* from the queue, but no more threads are allowed. This lock-free queue* is a fifo queue, the first element inserted is the first element which* comes out.** Thanks to Timur Doumler, Juce* https://www.youtube.com/watch?v=qdrp6k4rcP4*/#ifndef SPSCLOCKFREEQUEUE_H
#define SPSCLOCKFREEQUEUE_H#include <array>
#include <atomic>
#include <cassert>template <typename T, size_t fixedSize>
class SpScLockFreeQueue
{
public:///---------------------------------------------------------------------------/// @brief Constructor. Asserts when the underlying type is not lock freeSpScLockFreeQueue(){std::atomic<size_t> test;assert(test.is_lock_free());}SpScLockFreeQueue(const SpScLockFreeQueue& src) = delete;virtual ~SpScLockFreeQueue(){}///---------------------------------------------------------------------------/// @brief  Returns whether the queue is empty/// @return True when emptybool empty() const noexcept{bool isEmpty = false;const size_t readPosition = m_readPosition.load();const size_t writePosition = m_writePosition.load();if (readPosition == writePosition){isEmpty = true;}return isEmpty;}///---------------------------------------------------------------------------/// @brief  Pushes an element to the queue/// @param  element  The element to add/// @return True when the element was added, false when the queue is fullbool push(const T& element){const size_t oldWritePosition = m_writePosition.load();const size_t newWritePosition = getPositionAfter(oldWritePosition);const size_t readPosition = m_readPosition.load();if (newWritePosition == readPosition){// The queue is fullreturn false;}m_ringBuffer[oldWritePosition] = element;m_writePosition.store(newWritePosition);return true;}///---------------------------------------------------------------------------/// @brief  Pops an element from the queue/// @param  element The returned element/// @return True when succeeded, false when the queue is emptybool pop(T& element){if (empty()){// The queue is emptyreturn false;}const size_t readPosition = m_readPosition.load();element = std::move(m_ringBuffer[readPosition]);m_readPosition.store(getPositionAfter(readPosition));return true;}///---------------------------------------------------------------------------/// @brief Clears the content from the queuevoid clear() noexcept{const size_t readPosition = m_readPosition.load();const size_t writePosition = m_writePosition.load();if (readPosition != writePosition){m_readPosition.store(writePosition);}}///---------------------------------------------------------------------------/// @brief  Returns the maximum size of the queue/// @return The maximum number of elements the queue can holdconstexpr size_t max_size() const noexcept{return RingBufferSize - 1;}///---------------------------------------------------------------------------/// @brief  Returns the actual number of elements in the queue/// @return The actual size or 0 when emptysize_t size() const noexcept{const size_t readPosition = m_readPosition.load();const size_t writePosition = m_writePosition.load();if (readPosition == writePosition){return 0;}size_t size = 0;if (writePosition < readPosition){size = RingBufferSize - readPosition + writePosition;}else{size = writePosition - readPosition;}return size;}static constexpr size_t getPositionAfter(size_t pos) noexcept{return ((pos + 1 == RingBufferSize) ? 0 : pos + 1);}private:// A lock-free queue is basically a ring buffer.static constexpr size_t RingBufferSize = fixedSize + 1;std::array<T, RingBufferSize> m_ringBuffer;std::atomic<size_t> m_readPosition = {0};std::atomic<size_t> m_writePosition = {0};    
};#endif /* SPSCLOCKFREEQUEUE_H */

服务器初始化

在服务器初始化阶段,我们根据预定义的队列大小来初始化这些无锁队列。

ThreadPool(size_t queueSize = 10000) {pool_ = std::make_shared<Pool>();pool_->tasks.resize(/* 线程数量 */, SpScLockFreeQueue<std::function<void()>>(queueSize));// ...(其他初始化代码)
}

性能对比与分析

通过与使用锁的传统设计进行比较,我们发现这种无锁设计在高并发环境下具有更好的性能。具体而言,吞吐量提高了约20%。

结论

通过使用无锁工作队列,我们成功地规避了因多线程锁而导致的性能开销,并在高并发环境下实现了更高的吞吐量。这证明了无锁数据结构在Web服务器设计中具有巨大的应用潜力。 潜在的问题:这个webserver采用轮询法为工作线程分配读写任务,如果某个线程读取一个耗时特别长的函数,就容易过载(堆积太多任务不能处理,但是每个任务的任务队列是设置了一个阈值的,比如堆积2000个线程就不能再继续增加了)

参考资料

  1. 无锁数据结构
  2. 高性能Web服务器设计

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

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

相关文章

精确到区县级街道乡镇行政边界geojson格式矢量数据的获取拼接实现Echarts数据可视化大屏地理坐标信息地图的解决方案

在Echarts制作地理信息坐标地图时&#xff0c;最麻烦的就是街道乡镇级别的行政geojson的获取&#xff0c; 文件大小 788M 文件格式 .json格式&#xff0c;由于是大文件数据&#xff0c;无法直接使用记事本或者IDE编辑器打开&#xff0c;推荐Dadroit Viewer&#xff08;国外…

实用电脑软件

使用电脑软件/手机app时&#xff0c;可以多搜索&#xff0c;多查资料&#xff0c;找到适合自己的&#xff0c;好用的软件。 一、效率 uTools 效率神器 ditto 复制粘贴栏。支持多次复制粘贴。 typora markdow写作&#xff0c;文件夹侧栏。 quicker 提高效率。 everyth…

基于粒子群优化算法、鲸鱼算法、改进的淘沙骆驼模型算法(PSO/SSA/tGSSA)的微电网优化调度(Matlab代码实现)

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

1.5 计算机网络的类别

思维导图&#xff1a; 1.5.1 计算机网络的定义 我的笔记&#xff1a; #### 精确定义&#xff1a; 计算机网络没有统一的精确定义&#xff0c;但一种较为接近的定义是&#xff1a;计算机网络主要由一些通用的、可编程的硬件互连而成&#xff0c;这些硬件并非专门用来实现某一特…

【Redis】数据过期策略和数据淘汰策略

数据过期策略和淘汰策略 过期策略 Redis所有的数据结构都可以设置过期时间&#xff0c;时间一到&#xff0c;就会自动删除。 问题&#xff1a;大家都知道&#xff0c;Redis是单线程的&#xff0c;如果同一时间太多key过期&#xff0c;Redis删除的时间也会占用线程的处理时间…

EF Core报错:Error Number:-2146893019

appsettings.json中的连接字符串要添加上&#xff1a;TrustServerCertificatetrue; 所以这里的连接字符串为&#xff1a;Data SourceLAPTOP-61GDB2Q7\\SQLEXPRESS;Initial CatalogMvcMovie.Data;Persist Security InfoTrue;TrustServerCertificatetrue;User IDsa;Passwordroot…

刷穿力扣(1~30)

更好的阅读体验 \huge{\color{red}{更好的阅读体验}} 更好的阅读体验 1. 两数之和 哈希表遍历数组&#xff0c;同时用 HashMap 维护已出现过的数及其下标若当前的数 nums[i] 满足 target - nums[i] 曾经出现过&#xff0c;则直接返回否则将其加入到哈希表中。 class Solution …

51单片机+EC11编码器实现可调参菜单+OLED屏幕显示

51单片机+EC11编码器实现可调参菜单+OLED屏幕显示 📍相关篇《stc单片机使用外部中断+EC11编码器实现计数功能》 🎈《STC单片机+EC11编码器实现调节PWM输出占空比》 🌼实际操作效果 🍁整个项目实现框架: 📓EC11接线原理图: 📓项目工程简介 📝仅凭借一个EC11编…

《Attention Is All You Need》论文笔记

下面是对《Attention Is All You Need》这篇论文的浅读。 参考文献&#xff1a; 李沐论文带读 HarvardNLP 《哈工大基于预训练模型的方法》 下面是对这篇论文的初步概览&#xff1a; 对Seq2Seq模型、Transformer的概括&#xff1a; 下面是蒟蒻在阅读完这篇论文后做的一…

前端——Layui的导航栏与tab页联动

一、body <!-- 导航栏 --><div class"layui-side layui-bg-black"><div class"layui-side-scroll"><ul id"nav" class"layui-nav layui-nav-tree" lay-filter"stock"><li class"layui-n…

slamplay:用C++实现的SLAM工具集

0. 项目简介 slamplay 是一个功能强大的工具集合&#xff0c;可用于开始使用 C 来玩和试验 SLAM。这是一项正在进行的工作。它在单个 cmake 框架中安装并提供一些最重要的功能 后端框架&#xff08;g2o、gtsam、ceres、se-sync 等&#xff09;、 前端工具&#xff08;opencv、…

RK3568的CAN驱动适配

目录 背景&#xff1a; 1.内核驱动模块配置 2.设备树配置 3.功能测试 4.bug修复 背景&#xff1a; 某个项目上使用RK3568的芯片&#xff0c;需要用到4路CAN接口进行通信&#xff0c;经过方案评审后决定使用RK3568自带的3路CAN外加一路spi转的CAN实现功能&#xff0c;在这个…

SpringSecurity源码学习一:过滤器执行原理

目录 1. web过滤器Filter1.1 filter核心类1.2 GenericFilterBean1.3 DelegatingFilterProxy1.3.1 原理1.3.2 DelegatingFilterProxy源码 2. FilterChainProxy源码学习2.1 源码2.1.1 doFilterInternal方法源码2.1.1.1 getFilters()方法源码2.1.1.2 VirtualFilterChain方法源码 3…

Go复合类型之数组类型

Go复合类型之数组 文章目录 Go复合类型之数组一、数组(Array)介绍1.1 基本介绍1.2 数组的特点 二、数组的声明与初始化2.1 数组声明2.2 常见的数据类型声明方法2.3 数组的初始化方式一&#xff1a;使用初始值列表初始化数组方法二&#xff1a;根据初始值个数自动推断数组长度方…

【已解决】RuntimeError Java gateway process exited before sending its port number

RuntimeError: Java gateway process exited before sending its port number 问题 思路 &#x1f3af;方法一 在代码前加入如下代码&#xff08;如图&#xff09;&#xff1a; import os os.environ[‘JAVA_HOME’] “/usr/local/jdk1.8.0_221” # 记得把地址改成自己的 …

golang pg 数据库不存在 就创建 --chatPGT

问&#xff1a;linkOrCreateDatabase(addr ), 函数执行 连接 pg数据库&#xff0c;若数据库 不存在就创建 gpt: 要在 Go 中连接到 PostgreSQL 数据库并在数据库不存在时创建数据库&#xff0c;你可以使用 github.com/lib/pq 包以及 database/sql 包。以下是一个示例&#xff1…

多校联测11 模板题

题目大意 给你四个整数 n , m , s e e d , w n,m,seed,w n,m,seed,w&#xff0c;其中 n , m n,m n,m为两个多项式 A ( x ) ∑ i 0 n a i x i A(x)\sum\limits_{i0}^na_ix^i A(x)i0∑n​ai​xi和 B ( x ) ∑ i 0 m b i x i B(x)\sum\limits_{i0}^mb_ix^i B(x)i0∑m​bi​xi…

Glide源码分析

一&#xff0c;Glide一次完整的加载流程 下面的流程图是一次完整的使用Glide加载图片流程,时序图 二&#xff0c;Glide重要的类图 三&#xff0c;Glide加载图片 流程图

华为云云耀云服务器L实例评测|Ubuntu 22.04部署edusoho-ct企培版教程 | 支持华为云视频点播对接CDN加速

华为云云耀云服务器L实例评测&#xff5c;Ubuntu 22.04部署edusoho企培版教程 1、选择购买 华为云耀云服务器L实例 简单上云第一步 2、选择你要安装的操作系统&#xff0c;例如 Ubuntu 22.04 server 64bit 3、然后支付订单就行了 4、华为云云耀云服务器L实例创建好之后&#x…

【高级rabbitmq】

文章目录 1. 消息丢失问题1.1 发送者消息丢失1.2 MQ消息丢失1.3 消费者消息丢失1.3.1 消费失败重试机制 总结 2. 死信交换机2.1 TTL 3. 惰性队列3.1 总结&#xff1a; 4. MQ集群 消息队列在使用过程中&#xff0c;面临着很多实际问题需要思考&#xff1a; 1. 消息丢失问题 1.1…