Linux_网络项目_WEB服务器 处理服务器写入失败后sigpipe信号导致服务器崩溃退出问题,引入线程池缓解大量请求,服务器组件化重构,在线计算机业务测试

文章目录

  • 1. 处理服务器写入管道出错
  • 2. 引入线程池缓解大量请求导致服务器崩溃
    • 设计线程任务类
    • 单例线程池组件设计
  • 3.代码位置
  • 4. 在线计算机业务运行截图

1. 处理服务器写入管道出错

经过测试,服务器在读取报文时如果出错可以选择直接关闭这个TCP里链接来节省资源。这个问题好解决不在赘述

此外如果服务器使用CGI时,向管道写入数据失败时,服务器进程会收到sigpipe信号直接崩溃。所以这里选择直接忽视这个信号。

class HttpSever
{
private:int port;TcpSever *tcp_sever;bool states; // 标记服务器运行状态void InitSever(){// 信号SIGPIPE需要进行忽略,防止服务器写入管道时失败服务器崩溃signal(SIGPIPE, SIG_IGN);tcp_sever = TcpSever::GetInstance(port);}public:HttpSever(int _port = PORT){port = _port;tcp_sever = nullptr;states = true;InitSever();}~HttpSever() {}void Loop(){LOG(INFO, "---http sever loop begin---");int listen_socket = tcp_sever->GetLinstenSocket();while (states != false){struct sockaddr_in client; // 客户端信息socklen_t len = sizeof(client);int sock = accept(listen_socket, (struct sockaddr *)&client, &len);if (sock < 0){// 套接字监听失败continue;}LOG(INFO, "get a new link");int *_sock = new int(sock);pthread_t tid = 0;pthread_create(&tid, nullptr, Entrance::HanderReq, _sock);pthread_detach(tid); // 线程分离}}
};

2. 引入线程池缓解大量请求导致服务器崩溃

设计线程任务类

#pragma once#include <iostream>
#include "../Protocol.hpp"// 设计线程任务队列
class Task
{
private:int sock;CallBack callback; // 设置回调
public:Task() {}Task(int _sock){sock = _sock;}void ProcessOn(){callback(sock);}
};

服务器在收到链接时不创建线程,而是构建线程任务,然后将线程任务放入到线程任务队列上,最后线程池从任务队列上拿去任务处理即可,线程拿到任务后,通过任务里的CallBack回调函数执行不同的任务

线程任务CallBack设计紧贴前面重构前的代码,将Entrance类更改成CallBack即可,这里设计仿函数,线程回调函数通过()直接访问解析请求函数

// 线程工作入口
class CallBack
{
public:void operator()(int sock)//仿函数{HanderReq(sock);}// 处理HTTP请求static void HanderReq(int _sock){LOG(INFO, "http request hander begin");EndPoint *endpoint = new EndPoint(_sock);endpoint->ReadRequest();if (endpoint->Stop() != true){LOG(INFO, "recv success! build request begin");endpoint->BuildResponse();endpoint->SendResponse();}else{LOG(WARNING, "recv error! please try again");}delete endpoint;LOG(INFO, "http request hander end");}
};

单例线程池组件设计

ThreadPool

#pragma once
#include <iostream>
#include "task.hpp"
#include <queue>
#include <pthread.h>
#include "../log/log.hpp"
#define THREAD_NUM 6
// 单例模式
class ThreadPool
{
private:// 生产者消费者模型std::queue<Task> task_queue; // 临界资源int thread_count;            // 线程数bool stop;                   // 线程池是否停止pthread_mutex_t mutex;pthread_cond_t cond;static ThreadPool *instance;bool init_threadPool() // 初始化线程池{for (int i = 0; i < thread_count; i++){pthread_t thread_id;if (0 != pthread_create(&thread_id, NULL, thread_routine, this)){// 创建线程失败LOG(FATAL, "init threadpool error!");return false;}}LOG(INFO, "http sever init threadpool success");return true;}static void *thread_routine(void *args) // 线程执行函数 static 删除类的this指针,对象通过线程的参数传递{ThreadPool *pool = (ThreadPool *)args;while (!pool->stop){Task task;pthread_mutex_lock(&pool->mutex);while (pool->task_queue.empty()) // while 不能换成if{pool->thread_wait(); // 线程唤醒后一定占有互斥锁}task = pool->task_queue.front();pool->task_queue.pop();pthread_mutex_unlock(&pool->mutex);task.ProcessOn();}return NULL;}void thread_wait() // 任务队列无任务,线程休眠{pthread_cond_wait(&cond, &mutex);}void thread_wakeup() // 任务队列有任务,线程唤醒{pthread_cond_signal(&cond); // 唤醒一个线程即可}ThreadPool(int thread_num = THREAD_NUM){thread_count = thread_num;stop = false;pthread_mutex_init(&mutex, nullptr);pthread_cond_init(&cond, nullptr);init_threadPool();}ThreadPool(const ThreadPool &) = delete;public:static ThreadPool *GetInstance(){static pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER; // 静态锁直接初始化,不需要释放if (instance == nullptr){pthread_mutex_lock(&lock);if (instance == nullptr){instance = new ThreadPool();}pthread_mutex_unlock(&lock);}return instance;}~ThreadPool(){pthread_mutex_destroy(&mutex);pthread_cond_destroy(&cond);}void push_task(const Task &task){pthread_mutex_lock(&mutex);task_queue.push(task);pthread_mutex_unlock(&mutex);thread_wakeup();}bool is_stop(){return stop;}
};ThreadPool *ThreadPool::instance = nullptr;

HttpSever函数更换为线程池同时组件化

#pragma once
#include "./TcpSever/TcpSever.hpp"
#include "Protocol.hpp"
#include "./log/log.hpp"
#include <signal.h>
#include <pthread.h>
#include "./ThreadPool/task.hpp"
#include "./ThreadPool/threadpool.hpp"
#define PORT 8080class HttpSever
{
private:int port;TcpSever *tcp_sever;bool states; // 标记服务器运行状态ThreadPool threadpool;void InitSever(){// 信号SIGPIPE需要进行忽略,防止服务器写入管道时失败服务器崩溃signal(SIGPIPE, SIG_IGN);tcp_sever = TcpSever::GetInstance(port);}public:HttpSever(int _port = PORT){port = _port;tcp_sever = nullptr;states = true;InitSever();}~HttpSever() {}void Loop(){LOG(INFO, "---http sever loop begin---");int listen_socket = tcp_sever->GetLinstenSocket();while (states != false){struct sockaddr_in client; // 客户端信息socklen_t len = sizeof(client);int sock = accept(listen_socket, (struct sockaddr *)&client, &len);if (sock < 0){// 套接字监听失败continue;}LOG(INFO, "get a new link");// 构建任务Task task(sock);threadpool.push_task(task);}}
};

测试运行结果如下:线程池创建6个线程,初始化线程池正确。同时用户报文发送错误,日志级别WARING级别
在这里插入图片描述

3.代码位置

Github
Gitee

4. 在线计算机业务运行截图

在这里插入图片描述
在这里插入图片描述
除0错误等其他错误会导致子进程错误退出,服务器捕捉到后跳转到错误页面即404.html
在这里插入图片描述
在这里插入图片描述

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

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

相关文章

tomcat下载

首先我们去官网进行下载&#xff0c;选择对应的版本&#xff08;找到64-bit&#xff09; 官网地址&#xff08;Apache Tomcat - Apache Tomcat 9 Software Downloads&#xff09; 下载图标如下&#xff0c;这里选择9.0.87版本 下载后如下 再将环境变量配置一下 到这一步就按照…

【类脑智能】脑网络通信模型分类及量化指标(附思维导图)

脑网络通信模型分类及量化指标(附思维导图) 参考论文&#xff1a;Brain network communication_ concepts, models and applications 概念 脑网络通信模型是一种使用图论和网络科学概念来描述和量化大脑结构中信息传递的模型。这种模型可以帮助研究人员理解神经信号在大脑内…

macbook删除软件只需几次点击即可彻底完成?macbook删除软件没有叉 苹果笔记本MacBook电脑怎么卸载软件? cleanmymac x怎么卸载

在MacBook的使用过程中&#xff0c;软件安装和卸载是我们经常需要进行的操作。然而&#xff0c;不少用户在尝试删除不再需要的软件时&#xff0c;常常发现这个过程既复杂又耗时。尽管MacOS提供了一些基本的macbook删除软件方法&#xff0c;但很多时候这些方法并不能彻底卸载软件…

【数据结构和算法初阶(C语言)】二叉树铺垫--栈帧的创建与销毁--细节全解

前言&#xff1a; 学习这么久以来&#xff0c;可能有很多疑问&#xff1a;局部变量怎么创建的&#xff1f;为什么局部变量的值是随机的&#xff1f;函数是怎么传参的&#xff1f;传参的顺序是怎么样的&#xff1f;形参和实参是什么样的关系&#xff1f;函数调用是怎么做的&…

Vue | 使用 ECharts 绘制折线图

目录 一、安装和引入 ECharts 二、使用 ECharts 2.1 新增 div 盒子 2.2 编写画图函数 2.3 完整代码结构 三、各种小问题 3.1 函数调用问题 3.2 数据格式问题 3.3 坐标轴标签问题 3.4 间隔显示标签 参考博客&#xff1a;Vue —— ECharts实现折线图 本文是在上…

HTML5CSS3提高导读

HTML5CSS3提高导读 2024/2/20 HTML5 的新增特性主要是针对于以前的不足&#xff0c;增加了一些新的标签、新的表单和新的表单属性等。 这些新特性都有兼容性问题&#xff0c;基本是 IE9 以上版本的浏览器才支持&#xff0c;如果不考虑兼容性问题&#xff0c;可以大量使用这 …

面向对象(C# )

面向对象&#xff08;C# &#xff09; 文章目录 面向对象&#xff08;C# &#xff09;ref 和 out传值调用和引用调用ref 和 out 的使用ref 和 out 的区别 结构体垃圾回收GC封装成员属性索引器静态成员静态类静态构造函数拓展方法运算符重载内部类和分布类 继承里氏替换继承中的…

SPSS k-均值聚类的 anova分析表解读

from&#xff1a;SPSS K均值聚类&#xff08;k-means&#xff09;和可视化方法 - CollinsLi - 博客园 (cnblogs.com) F值&#xff1a;变量对聚类的贡献 显著性水平&#xff1a;<0.05 则因子显著

vanna:基于RAG的text2sql框架

文章目录 vanna简介及使用vanna的原理vanna的源码理解总结参考资料 vanna简介及使用 vanna是一个开源的利用了RAG的SQL生成python框架&#xff0c;在2024年3月已经有了5.8k的star数。 Vanna is an MIT-licensed open-source Python RAG (Retrieval-Augmented Generation) fram…

springboot趵突泉景区的智慧导游小程序

摘 要 当今社会已经步入了科学技术进步和经济社会快速发展的新时期&#xff0c;国际信息和学术交流也不断加强&#xff0c;计算机技术对经济社会发展和人民生活改善的影响也日益突出&#xff0c;人类的生存和思考方式也产生了变化。传统趵突泉景区的智慧导游采取了人工的管理方…

linux源配置:ubuntu、centos

1、ubuntu源配置 1&#xff09;先查电脑版本型号: lsb_release -c2&#xff09;再编辑源更新&#xff0c;源要与上面型号对应 参考&#xff1a;https://midoq.github.io/2022/05/30/Ubuntu20-04%E6%9B%B4%E6%8D%A2%E5%9B%BD%E5%86%85%E9%95%9C%E5%83%8F%E6%BA%90/ /etc/apt/…

Django 解决新建表删除后无法重新创建等问题

Django 解决新建表删除后无法重新创建等问题 问题发生描述处理办法首先删除了app对应目录migrations下除 __init__.py以外的所有文件:然后&#xff0c;删除migrations中关于你的app的同步数据数据库记录最后&#xff0c;重新执行迁移插入 问题发生描述 Django创建的表&#xf…