基于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://m.bishedaima.com/yuanma/36074.html

相关推荐

  • 基于SpringBoot框架的医疗挂号管理系统

    这是一套采用Java编程语言,基于SpringBoot框架构建的医疗预约管理系统源代码,项目中融入了Vue技术,开发工具为Idea或Eclipse,此系统适用于毕业设计或课程实践
    2024年05月23日
    2 1 1
  • 接管裸机的控制权之Python

    接管裸机的控制权 操作系统实验报告 实验题目 搭建和应用实验环境 接管裸机的控制权 实验目的 安装虚拟机软件 VMware
    2024年05月14日
    3 1 2
  • 基于SpringBoot框架的公司日常考勤系统

    这是一套采用Java语言,基于SpringBoot框架构建的日常考勤管理系统源代码,🔥🔥项目中融入了SpringBoot和Vue技术栈,适用于开发环境如Idea或Eclipse
    2024年05月23日
    3 1 1
  • 基于Java Swing实现的简易考试系统界面搭建

    基于Java Swing实现的简易考试系统界面搭建 一,任务介绍 1,1 任务描述 本任务要搭建一个考试系统界面,考试题型主要有两类:判断题和单选题
    2024年05月14日
    2 1 1
  • 高校心理教育辅导平台

    这是一个🔥🔥基于SpringBoot框架的高校心理教育辅导平台设计与实现🔥🔥的项目源码,开发语言Java,框架使用的SpringBoot+vue技术,开发环境Idea/Eclipse
    2024年05月23日
    4 1 1
  • 基于安卓的备忘录应用开发实现

    这是一个🔥🔥基于安卓的备忘录应用开发实现🔥🔥的项目源码,开发语言Java,开发环境Idea/Eclipse,这个 安卓备忘录应用开发技术栈为安卓项目,可以作为毕业设计课程设计作业基于安卓平台实现一个日记本备忘录应用
    2024年05月23日
    1 1 1
  • 实现一个轻量级的 Web 服务器

    实现一个轻量级的 Web 服务器 实验目的 深入掌握 HTTP 协议规范,学习如何编写标准的互联网应用服务器, 实验内容 服务程序能够正确解析 HTTP 协议
    2024年05月14日
    2 1 1
  • 基于Springmvc+MyBatis+Spring+Bootstrap+EasyUI+Mysql的个人博客系统

    Blog-master 基于Springmvc+MyBatis+Spring+Bootstrap+EasyUI+Mysql的个人博客系统 参考文献 基于SSH框架的博客用户分享平台的设计与实现(河北工业大学·刘磊) 基于SSH框架的人力资源管理系统的设计与开发(电子科技大学·唐伟) 博客管理系统的设计与实现(吉林大学·赵岩) 基于SSH架构的个人空间交友网站的设计与实现(北京邮电大学·隋昕航) 基于MVC设计模式的博客系统的设计与实现(大连理工大学·侯林) 基于SSH框架的博客用户分享平台的设计与实现(河北工业大学·刘磊) 基于SSH框架模式的博客系统的设计与实现(西北师范大学·王刚成) 基于SSH2的轻博客系统的研究与实现(吉林大学·杨雪梅) 博客管理系统的设计与实现(吉林大学·赵岩) 基于OAuth2
    2024年05月14日
    4 1 1
  • 基于SSM框架的快递管理系统源码

    这是一个🔥🔥基于SSM框架的快递管理系统源码🔥🔥的项目源码,开发语言Java,开发环境Idea/Eclipse,这个 快递管理系统开发技术栈为SSM项目,可以作为毕业设计课程设计作业使用SSM框架(springmvc+spring+mybatis)实现一个JavaWeb快递管理系统
    2024年05月23日
    13 1 2
  • 基于JSP的房屋出租管理系统设计与实现

    基于JSP的房屋出租管理系统设计与实现 摘 要 本文研究并实现了房屋出租管理系统,主要用运了servlet技术,java 技术,jsp网页编程技术,jQuery EasyUI界面美化插件
    2024年05月14日
    20 1 1

发表回复

登录后才能评论