基于C++实现的简易WEB服务器

C++实现的简易WEB服务器 1 项目描述: 一个基于c++实现的web服务器,可以对http请求进行解析响应,可以支持上万的QPS 2 项目环境: linux

本文包含相关资料包-----> 点击直达获取<-------

C++实现的简易WEB服务器

1 项目描述:

一个基于c++实现的web服务器,可以对http请求进行解析响应,可以支持上万的QPS

2 项目环境:

linux,c++14,MySql

3 项目要点:

  1. 实现了基于epoll和线程池的多线程Reactor高并发模型;

  2. 实现了基于小根堆实现的计时器,用来关闭超时的停止活动的客户端连接;

  3. 实现了基于单例和阻塞队列的异步日志系统,用来记录服务器的运行状态;

  4. 实现了基于RAII机制的数据库连接池,用来减少数据库连接建立和关闭的开销。

  5. 使用标准库容器封装char,用来实现缓冲区的自动增长;

  6. 使用正则表达式和状态机来解析HTTP请求消息,用来实现对静态资源的请求。

4 项目细节

4.1 工作逻辑

main函数中先初始化服务器WebServer对象server,调用 server.Start(); 函数启动服务器。

WebServer类中启动服务器的start函数如下:

函数循环调用epoll来监听连接事件,写事件和读事件,并对这些事件进行相应的处理,对于监听到的异常事件,服务器关闭相关的客户端连接。

c++ /* 启动服务器 */ void WebServer::Start() { int timeMS = -1; // epoll wait timeout == -1 无事件将阻塞 if(!isClose_) { LOG_INFO("========== Server start =========="); } // 循环调用epoll while(!isClose_) { // 解决超时连接 if(timeoutMS_ > 0) { timeMS = timer_->GetNextTick(); // 清除超时的客户端连接,并得到下一次超时的时间 } int eventCnt = epoller_->Wait(timeMS); // 使得epoller_wait阻塞超过timeMS时间返回 for(int i = 0; i < eventCnt; i++) { /* 处理事件 */ int fd = epoller_->GetEventFd(i); uint32_t events = epoller_->GetEvents(i); if(fd == listenFd_) { //处理监听事件 DealListen_(); } else if(events & (EPOLLRDHUP | EPOLLHUP | EPOLLERR)) { //异常 assert(users_.count(fd) > 0); CloseConn_(&users_[fd]); } else if(events & EPOLLIN) { //处理读操作 assert(users_.count(fd) > 0); DealRead_(&users_[fd]); } else if(events & EPOLLOUT) { //处理写操作 assert(users_.count(fd) > 0); DealWrite_(&users_[fd]); } else { LOG_ERROR("Unexpected event"); } } } }

但监听到连接事件的时候,程序调用 DealListen_ 函数对连接进行处理,函数具体如下,程序调用accept函数建立与客户端的连接,在ET模式模式下我们需要一次性处理所有的连接。

```c++ / 处理连接事件 / void WebServer::DealListen_() { struct sockaddr_in addr; //保存连接的客户端的地址 socklen_t len = sizeof(addr); do { //非阻塞的accept int fd = accept(listenFd_, (struct sockaddr *)&addr, &len);

    if(fd <= 0) { return;}
    else if(HttpConn::userCount >= MAX_FD) {
        SendError_(fd, "Server busy!");
        LOG_WARN("Clients is full!");
        return;
    }

    AddClient_(fd, addr);            //添加客户端
} while(listenEvent_ & EPOLLET);     //ET模式:需要一次性处理全部的连接

} ```

AddClient_函数将每一个客户端连接都封装成为HttpConn类,并给这个客户端注册写事件,使得epoll可以监听到该客户端的请求(HttpConn类中的文件描述符的读事件)。

c++ void WebServer::AddClient_(int fd, sockaddr_in addr) { assert(fd > 0); users_[fd].init(fd, addr); if(timeoutMS_ > 0) { // 超时调用回调函数,关闭连接 timer_->add(fd, timeoutMS_, std::bind(&WebServer::CloseConn_, this, &users_[fd])); } // 给这个客户端注册写事件 epoller_->AddFd(fd, EPOLLIN | connEvent_); SetFdNonblock(fd); LOG_INFO("Client[%d] in!", users_[fd].GetFd()); }

但监听到读事件的时候,程序调用 DealRead_ 函数对读时间进行处理,传入的参数为epoll监听到写事件的文件描述符对应的HttpConn,函数具体如下,程序将读的工作(OnRead_工作函数)添加到线程池中的工作队列中,交给线程池中的线程去处理。

c++ /* 处理读事件 */ void WebServer::DealRead_(HttpConn* client) { assert(client); ExtentTime_(client); // 延长超时时间 //由线程池中的子线程来处理读操作 threadpool_->AddTask(std::bind(&WebServer::OnRead_, this, client)); }

OnRead_工作函数如下,程序调用对应客户端连接的read函数,将客户端连接的套接字中读数据到改客户端连接的对应的readBuff_中。

c++ /* 线程池中的子线程的读工作函数 */ void WebServer::OnRead_(HttpConn* client) { assert(client); int ret = -1; int readErrno = 0; ret = client->read(&readErrno); // 读取客户端套接字的数据,读到httpconn的读缓存区 if(ret <= 0 && readErrno != EAGAIN) { // 读异常就关闭客户端 CloseConn_(client); return; } // 业务逻辑的处理(先读后处理) OnProcess(client); }

c++ ssize_t HttpConn::read(int* saveErrno) { ssize_t len = -1; do { // 客户端的套接字中读数据到readBuff_中 len = readBuff_.ReadFd(fd_, saveErrno); if (len <= 0) { break; } } while (isET);//ET:要一次性全部读出 return len; }

当读事件完成以后,需要调用OnProcess函数对读到的数据进行处理(对读到的请求数据进行解析和响应)。

c++ /* 处理读(请求)数据的函数 */ void WebServer::OnProcess(HttpConn* client) { if(client->process()) { epoller_->ModFd(client->GetFd(), connEvent_ | EPOLLOUT);//相应成功,修改监听事件为写 } else { epoller_->ModFd(client->GetFd(), connEvent_ | EPOLLIN); } }

参考文献

  • 基于.NET架构的商业网站设计与实现(山东大学·张超)
  • 基于J2EE的远程教育平台的开发与实现(吉林大学·葛瑛)
  • 基于ASP.NET开发技术的BBS论坛研究与设计(中国海洋大学·马章勤)
  • 基于Qt的跨平台web服务开发框架(西安电子科技大学·张劲峰)
  • 基于.NET平台的XML Web Services研究与实现(兰州理工大学·杨静)
  • 中小型物流企业物流信息平台设计与实现(西安电子科技大学·孙瀚渝)
  • 基于JSP的雄霸天下游戏网的后台操作系统的开发设计(电子科技大学·张璇)
  • 利用.NET技术实现电子政务系统的研究(武汉理工大学·甘俊)
  • 基于Web服务的企业信息化集成应用(云南财经大学·韩雪)
  • 基于.NET下Web服务的信息查询系统的研究与设计(合肥工业大学·张静)
  • 基于SSH架构的个人空间交友网站的设计与实现(北京邮电大学·隋昕航)
  • 基于.NET架构的商业网站设计与实现(山东大学·张超)
  • 基于.NET架构的商业网站设计与实现(山东大学·张超)
  • 基于B/S架构的DCS现场控制站数据服务器设计(河北大学·张钊熙)
  • 基于WEB技术的新闻发布系统的设计与实现(电子科技大学·黄红)

本文内容包括但不限于文字、数据、图表及超链接等)均来源于该信息及资料的相关主题。发布者:源码码头网 ,原文地址:https://bishedaima.com/yuanma/36074.html

相关推荐

  • 基于Python实现聚类算法

    一,理论知识 1,1 K-Means 给定一组数据集,聚类算法将它们分成不同的子组,我们希望类内实例高度相似,类间实例低相似, 在样本集中,随机选取K个点作为中心$\bold \mu_k$
    2024年05月14日
    2 1 1
  • 图像去雾Python

    图像去雾 一,总述 本次大作业要求调研实现去雾算法,发现其中的问题,并对算法进行改进, 我首先实现了基于暗原色先验的去雾算法,并从运算速度和去雾效果方面进行了一定的改进
    2024年05月14日
    2 1 1
  • 基于Python实现的新闻网络爬虫程序

    基于Python实现的新闻网络爬虫程序 1,简介 1,1 引用术语与缩写解释 缩写,术语 解 释 Python 一种简洁而强大的解释型脚本语言 pyodbc Python下的ODBC数据库访问组件 SQLAlchemy Python下的ORM数据访问组件 pywin32 Python下的Win32接口访问组件 requests Python下的Web访问组件 Pillow Python下的图像处理组件 解释型语言 无需编译源码可敏捷部署并执行的语言 IOC 控制反转
    2024年05月14日
    4 1 2
  • Java+Swing实现仿windows计算器程序

    这是一个🔥🔥Swing实现仿windows计算器程序🔥🔥的项目源码,开发语言Java,开发环境Idea/Eclipse,这个 计算器程序开发技术栈为JavaSwing项目
    2024年05月23日
    1 1 1
  • 基于Springboot+RabbitMQ+redis实现的超市订单管理系统

    基于Springboot+RabbitMQ+redis实现的超市订单管理系统 1 系统需求分析 超市订单管理系统是一个专为连锁店,超市等商业场所提供订单管理平台的系统
    2024年05月14日
    5 1 2
  • 基于springboot开发短视频网站

    基于springboot的短视频网站的开发与设计 前言 毕业设计 使用的开发工具是ecplise,idea的话自己新建个项目copy一下,使用了springboot
    2024年05月14日
    4 1 1
  • 基于SpringAop和ThreadLocal实现的收集与处理请求方法埋点信息的日志工具包

    基于SpringAop和ThreadLocal实现的收集与处理请求方法埋点信息的日志工具包 1,项目简介 设计目的和场景 : 使用Spring AOP拦截方法参数大部分做法基本上大同小异
    2024年05月14日
    3 1 1
  • 基于SpringBoot和MySQL的社区物业管理系统

    基于SpringBoot和MySQL的社区物业管理系统 1,项目简介 1,1 介绍 基于SpringBoot+Layui的社区物业管理系统 分为业主和管理员两个角色: 业主 车位费查询 物业费查询 我的投诉列表 我的报修列表等功能模块 管理员 车位收费管理 物业收费管理 投诉信息管理 报修信息管理 楼宇管理 房屋管理 业主管理 车位管理 抄表管理 用户管理 报表分析等功能 1
    2024年05月14日
    14 1 2
  • 基于Java EE和MySQL的平台项目管理系统的设计与实现

    基于Java EE和MySQL的平台项目管理系统的设计与实现 摘 要 随着社会信息化的发展,很多的社会管理问题也一并出现了根本性变化,项目公司的报表及文件管理也发生了变化
    2024年05月14日
    2 1 1
  • 基于Spring开发轻量级分布式 RPC 框架

    🏆 从零开始设计一个轻量级分布式 RPC 框架 💌 写在前面 本项目基于 Spring + Netty + Zookeeper + Protostuff 从零开始设计实现一个轻量级的分布式 RPC 框架
    2024年05月14日
    2 1 1

发表回复

登录后才能评论