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

TinyWebSever源码逐行注释()_简单代码的整合

前言

项目源码地址
项目详细介绍

项目简介:
Linux下C++轻量级Web服务器,助力初学者快速实践网络编程,搭建属于自己的服务器.

  1. 使用 线程池 + 非阻塞socket + epoll(ET和LT均实现) + 事件处理(Reactor和模拟Proactor均实现) 的并发模型
  2. 使用状态机解析HTTP请求报文,支持解析GET和POST请求
  3. 访问服务器数据库实现web端用户注册、登录功能,可以请求服务器图片和视频文件
  4. 实现同步/异步日志系统,记录服务器运行状态
  5. 经Webbench压力测试可以实现上万的并发连接数据交换
    以下是该项目中比较简单的代码的源码注释:

main.cpp

#include "config.h"int main(int argc, char *argv[])
{//需要修改的数据库信息,登录名,密码,库名string user = "root";string passwd = "root";string databasename = "qgydb";//命令行解析Config config;config.parse_arg(argc, argv);WebServer server;//初始化server.init(config.PORT, user, passwd, databasename, config.LOGWrite, config.OPT_LINGER, config.TRIGMode,  config.sql_num,  config.thread_num, config.close_log, config.actor_model);//日志server.log_write();//数据库server.sql_pool();//线程池server.thread_pool();//触发模式server.trig_mode();//监听server.eventListen();//运行server.eventLoop();return 0;
}

webserver.h

#ifndef WEBSERVER_H
#define WEBSERVER_H// 导入必要的头文件,用于处理网络编程、文件描述符、epoll机制、线程池等功能
#include <sys/socket.h>  // 套接字相关
#include <netinet/in.h>  // IP地址相关
#include <arpa/inet.h>   // IP转换相关
#include <stdio.h>       // 标准输入输出
#include <unistd.h>      // POSIX 操作系统 API,如close等
#include <errno.h>       // 错误处理
#include <fcntl.h>       // 文件控制(非阻塞设置)
#include <stdlib.h>      // 标准库函数,如malloc等
#include <cassert>       // 断言,用于调试
#include <sys/epoll.h>   // epoll相关// 导入线程池和HTTP连接相关模块
#include "./threadpool/threadpool.h"
#include "./http/http_conn.h"// 常量定义
const int MAX_FD = 65536;           // 最大文件描述符数量
const int MAX_EVENT_NUMBER = 10000; // epoll 监听的最大事件数
const int TIMESLOT = 5;             // 定时器的最小超时时间单位(秒)class WebServer
{
public:// 构造函数WebServer();// 析构函数~WebServer();// 初始化函数,设置服务器端口、数据库信息、日志等void init(int port , string user, string passWord, string databaseName,int log_write , int opt_linger, int trigmode, int sql_num,int thread_num, int close_log, int actor_model);// 初始化线程池void thread_pool();// 初始化数据库连接池void sql_pool();// 初始化日志系统void log_write();// 设置触发模式(边沿触发或水平触发)void trig_mode();// 设置监听事件(epoll监听)void eventListen();// 事件循环,处理服务器运行中的各类事件void eventLoop();// 处理定时器的回调,管理客户端连接void timer(int connfd, struct sockaddr_in client_address);// 调整定时器的时间void adjust_timer(util_timer *timer);// 处理超时的定时器,关闭连接void deal_timer(util_timer *timer, int sockfd);// 处理客户端数据的读取事件bool dealclientdata();// 处理信号(如关闭服务器、超时等)bool dealwithsignal(bool& timeout, bool& stop_server);// 处理读事件void dealwithread(int sockfd);// 处理写事件void dealwithwrite(int sockfd);public:// 基础参数int m_port;             // 服务器端口号char *m_root;           // 服务器根目录int m_log_write;        // 日志写入方式int m_close_log;        // 是否关闭日志int m_actormodel;       // 事件处理模式(Reactor/Proactor)// 管道文件描述符(用于处理信号)int m_pipefd[2];// epoll 文件描述符int m_epollfd;// 所有客户端的 HTTP 连接数据http_conn *users;// 数据库相关connection_pool *m_connPool;    // 数据库连接池string m_user;         // 数据库用户名string m_passWord;     // 数据库密码string m_databaseName; // 数据库名称int m_sql_num;         // 数据库连接数量// 线程池相关threadpool<http_conn> *m_pool; // 线程池指针int m_thread_num;              // 线程数量// epoll 事件相关epoll_event events[MAX_EVENT_NUMBER]; // 用于存储epoll等待到的事件// 套接字相关int m_listenfd;        // 监听套接字int m_OPT_LINGER;      // 是否使用优雅关闭连接(linger选项)int m_TRIGMode;        // 触发模式(边沿或水平触发)int m_LISTENTrigmode;  // 监听套接字的触发模式int m_CONNTrigmode;    // 连接套接字的触发模式// 定时器相关client_data *users_timer; // 用户定时器数据Utils utils;              // 工具类,管理定时器和信号
};
#endif

Config.cpp

这里面也只有一个解析命令行的代码:

void Config::parse_arg(int argc, char*argv[]){int opt;const char *str = "p:l:m:o:s:t:c:a:";while ((opt = getopt(argc, argv, str)) != -1){switch (opt){case 'p':{PORT = atoi(optarg);break;}case 'l':{LOGWrite = atoi(optarg);break;}case 'm':{TRIGMode = atoi(optarg);break;}case 'o':{OPT_LINGER = atoi(optarg);break;}case 's':{sql_num = atoi(optarg);break;}case 't':{thread_num = atoi(optarg);break;}case 'c':{close_log = atoi(optarg);break;}case 'a':{actor_model = atoi(optarg);break;}default:break;}}
}

这段代码是 Config::parse_arg 函数的实现,它用于解析命令行参数,将传入的命令行参数根据不同的标志(如 -p, -l, -m 等)转换为对应的配置值。

下面是详细的解释:

1. 函数签名

void Config::parse_arg(int argc, char *argv[])
  • 这是一个 Config 类的成员函数,名字是 parse_arg
  • argc 是命令行参数的个数,argv 是存储命令行参数的字符数组。argcargv 通常在 main 函数中作为参数传递。
  • 这个函数的作用是根据命令行输入的参数,解析并设置类中的相关配置项。

2. 定义变量

int opt;
const char *str = "p:l:m:o:s:t:c:a:";
  • opt:用于存储解析到的选项字符(如 -p, -l 等)。
  • str:定义了命令行参数选项的格式。每个字符代表一个参数的标志,后面的冒号(:)表示该选项需要一个参数。例如,'p' 后面有冒号,因此 -p 选项必须带有一个参数。

3. getopt 函数

while ((opt = getopt(argc, argv, str)) != -1)
  • getopt 是一个用于解析命令行参数的标准库函数,它依次解析由 argv 传入的参数,并根据 str 中定义的选项返回对应的标志字符(如 pl 等)。
  • getopt 返回 -1 时,表示已经没有更多的选项可供处理。

4. switch 语句处理选项

  • 根据 getopt 返回的标志字符(存储在 opt 中),switch 语句分别处理不同的命令行选项。
switch (opt)
{case 'p':PORT = atoi(optarg);break;// 其他选项处理
}
  • 每个 case 语句处理对应的标志选项(如 -p-l 等)。
  • atoi(optarg)optarggetopt 提供的当前选项的参数值(它是一个字符串)。atoi 函数用于将字符串转换为整数。例如,当用户输入 -p 8080 时,optarg"8080",而 atoi(optarg) 将其转换为整数 8080
  • 这些参数分别赋值给类中的成员变量(如 PORT, LOGWrite, TRIGMode 等)。

5. 解析的命令行选项

  • -p:解析端口号,赋值给 PORT
  • -l:日志写入方式,赋值给 LOGWrite
  • -m:触发模式,赋值给 TRIGMode
  • -o:设置 linger 选项,赋值给 OPT_LINGER
  • -s:数据库连接池的连接数,赋值给 sql_num
  • -t:线程池中的线程数,赋值给 thread_num
  • -c:是否关闭日志,赋值给 close_log
  • -a:选择处理模型,赋值给 actor_model

6. 默认行为

  • 如果传入了未定义的选项(即不在 str 中的标志),default 部分将不做任何处理。

Config.h

#ifndef CONFIG_H
#define CONFIG_H#include "webserver.h"using namespace std;class Config
{
public:Config();~Config(){};void parse_arg(int argc, char*argv[]);//端口号int PORT;//日志写入方式int LOGWrite;//触发组合模式int TRIGMode;//listenfd触发模式int LISTENTrigmode;//connfd触发模式int CONNTrigmode;//优雅关闭链接int OPT_LINGER;//数据库连接池数量int sql_num;//线程池内的线程数量int thread_num;//是否关闭日志int close_log;//并发模型选择int actor_model;
};#endif

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 将 x 减到 0 的最小操作数
  • 视觉语言模型(VLMs)知多少?
  • 【2024】MySQL库表基本操作
  • 从 CRX 文件安装 Chrome 扩展程序
  • 安卓13带有系统签名的应用不能正常使用webview 调用webview失败 系统应用app apk
  • 突破代码:克服编程学习中的挫折感
  • 创建一个Oracle版本的JDK的Docker镜像
  • C++笔记15•数据结构:二叉树之二叉搜索树•
  • STM32 系列MCU 开发利器 STM32CubeIDE
  • Pandas_[‘index‘]_is_not_found_in_axis
  • 解读:以RTC为基,AI为脑的“超拟人”AI实时互动解决方案
  • 【IEEE出版,IEEE Xplore等多数据库检索】第五届智能设计国际会议(ICID 2024,10月25-27)
  • Excel如何把表格变成图表
  • 什么叫做 “沿着晶体平面偏析”
  • 优质的产业园都在怎么做运营?
  • 《网管员必读——网络组建》(第2版)电子课件下载
  • 【笔记】你不知道的JS读书笔记——Promise
  • 【前端学习】-粗谈选择器
  • 2019.2.20 c++ 知识梳理
  • CAP 一致性协议及应用解析
  • Docker入门(二) - Dockerfile
  • ECMAScript入门(七)--Module语法
  • Laravel 中的一个后期静态绑定
  • mysql 5.6 原生Online DDL解析
  • MySQL数据库运维之数据恢复
  • Promise初体验
  • PyCharm搭建GO开发环境(GO语言学习第1课)
  • React16时代,该用什么姿势写 React ?
  • SpringCloud(第 039 篇)链接Mysql数据库,通过JpaRepository编写数据库访问
  • Spring框架之我见(三)——IOC、AOP
  • Spring声明式事务管理之一:五大属性分析
  • vue脚手架vue-cli
  • 给自己的博客网站加上酷炫的初音未来音乐游戏?
  • 技术胖1-4季视频复习— (看视频笔记)
  • 利用阿里云 OSS 搭建私有 Docker 仓库
  • 学习JavaScript数据结构与算法 — 树
  • 自制字幕遮挡器
  • AI又要和人类“对打”,Deepmind宣布《星战Ⅱ》即将开始 ...
  • 曜石科技宣布获得千万级天使轮投资,全方面布局电竞产业链 ...
  • #我与Java虚拟机的故事#连载05:Java虚拟机的修炼之道
  • #预处理和函数的对比以及条件编译
  • (~_~)
  • (12)Linux 常见的三种进程状态
  • (3)选择元素——(14)接触DOM元素(Accessing DOM elements)
  • (4) openssl rsa/pkey(查看私钥、从私钥中提取公钥、查看公钥)
  • (web自动化测试+python)1
  • (笔试题)分解质因式
  • (二)【Jmeter】专栏实战项目靶场drupal部署
  • (附源码)spring boot基于小程序酒店疫情系统 毕业设计 091931
  • (论文阅读笔记)Network planning with deep reinforcement learning
  • (牛客腾讯思维编程题)编码编码分组打印下标题目分析
  • (三)Hyperledger Fabric 1.1安装部署-chaincode测试
  • (四)activit5.23.0修复跟踪高亮显示BUG
  • (四)linux文件内容查看
  • (一)python发送HTTP 请求的两种方式(get和post )