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

2024-09-05 20:20

本文主要是介绍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

这篇关于TinyWebSever源码逐行注释()_简单代码的整合的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1139964

相关文章

Python中注释使用方法举例详解

《Python中注释使用方法举例详解》在Python编程语言中注释是必不可少的一部分,它有助于提高代码的可读性和维护性,:本文主要介绍Python中注释使用方法的相关资料,需要的朋友可以参考下... 目录一、前言二、什么是注释?示例:三、单行注释语法:以 China编程# 开头,后面的内容为注释内容示例:示例:四

SpringBoot整合liteflow的详细过程

《SpringBoot整合liteflow的详细过程》:本文主要介绍SpringBoot整合liteflow的详细过程,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋...  liteflow 是什么? 能做什么?总之一句话:能帮你规范写代码逻辑 ,编排并解耦业务逻辑,代码

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

springboot整合TDengine全过程

《springboot整合TDengine全过程》:本文主要介绍springboot整合TDengine全过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录环境准备JDBC-JNI方式准备依赖实体类Mapper配置类测试类RESTful方式实体类配置类测试类总结

MySQL数据库的内嵌函数和联合查询实例代码

《MySQL数据库的内嵌函数和联合查询实例代码》联合查询是一种将多个查询结果组合在一起的方法,通常使用UNION、UNIONALL、INTERSECT和EXCEPT关键字,下面:本文主要介绍MyS... 目录一.数据库的内嵌函数1.1聚合函数COUNT([DISTINCT] expr)SUM([DISTIN

Java实现自定义table宽高的示例代码

《Java实现自定义table宽高的示例代码》在桌面应用、管理系统乃至报表工具中,表格(JTable)作为最常用的数据展示组件,不仅承载对数据的增删改查,还需要配合布局与视觉需求,而JavaSwing... 目录一、项目背景详细介绍二、项目需求详细介绍三、相关技术详细介绍四、实现思路详细介绍五、完整实现代码

Go语言代码格式化的技巧分享

《Go语言代码格式化的技巧分享》在Go语言的开发过程中,代码格式化是一个看似细微却至关重要的环节,良好的代码格式化不仅能提升代码的可读性,还能促进团队协作,减少因代码风格差异引发的问题,Go在代码格式... 目录一、Go 语言代码格式化的重要性二、Go 语言代码格式化工具:gofmt 与 go fmt(一)

HTML5实现的移动端购物车自动结算功能示例代码

《HTML5实现的移动端购物车自动结算功能示例代码》本文介绍HTML5实现移动端购物车自动结算,通过WebStorage、事件监听、DOM操作等技术,确保实时更新与数据同步,优化性能及无障碍性,提升用... 目录1. 移动端购物车自动结算概述2. 数据存储与状态保存机制2.1 浏览器端的数据存储方式2.1.

基于 HTML5 Canvas 实现图片旋转与下载功能(完整代码展示)

《基于HTML5Canvas实现图片旋转与下载功能(完整代码展示)》本文将深入剖析一段基于HTML5Canvas的代码,该代码实现了图片的旋转(90度和180度)以及旋转后图片的下载... 目录一、引言二、html 结构分析三、css 样式分析四、JavaScript 功能实现一、引言在 Web 开发中,