Linux C++项目推荐:文件服务器+如何快速上手C++大项目

发布于:2025-02-28 ⋅ 阅读:(14) ⋅ 点赞:(0)

1 本次分享的意义

源码地址:https://github.com/shangguanyongshi/WebFileServer.git

视频讲解:Linux C++项目推荐:WebFileServer文件服务器+如何快速上手C++大项目

学完C++和Linux编程后, 可以以WebFileServer文件服务器作为练手项目,比WebServer项目多了文件上传的功能,更加有意义。

这次分享不单只是讲解这个WebFileServer项目如何运行,也能让大家掌握如何快速掌握自己不熟悉的C++项目。

2 项目功能

Web 文件服务器,通过浏览器发送 HTTP 请求管理服务器指定文件夹下的所有文件。主要功能包括:

  • 以 HTML 页面形式返回该文件夹下的所有文件

  • 可以选择本地文件上传到服务器

  • 可以对列表中的文件文件执行下载操作

  • 可以删除服务器中的指定文件

3 整体框架

  • 使用 Reactor 事件处理模型,通过统一事件源,主线程使用 epoll 监听所有的事件,工作线程负责执行事件的逻辑处理

  • 预先创建线程池,当有事件发生时,加入线程池的工作队列中,使用随机选择算法选择线程池中的一个线程处理工作队列的事件

  • 使用 HTTP GET 方法获取文件列表,发起下载文件、删除文件的请求。使用 POST 方法向服务器上传文件

  • 服务端使用有限状态机对请求消息进行解析,根据解析结果执行操作后,向客户端发送页面、发送文件或发送重定向报文

  • 服务端使用 sendfile 函数实现零拷贝数据发送

4 编译和运行

1.build 项目

sh ./build.sh

默认的编译不支持调试,可以修改makefile增加-g编译选项。

然后再重新编译,以方便调试代码。

2.启动文件服务器

./main

3.在浏览器端输入 服务端ip:端口号(端口号默认是8888)

serverip:8888

5 如何快速分析代码

5.1 main函数入口

(gdb) b main
Breakpoint 1 at 0x5555555590ad: file main.cpp, line 3.

5.2 分析网络模型

通过info threads可以看到 共有五个线程,其中主线程是调用了epoll,再加上README的讲解可以推断出来这个项目是单个epoll +线程池的模式

* 1    Thread 0x7ffff79a9740 (LWP 339328) "main" 0x00007ffff7c1c68e in epoll_wait (epfd=4, events=0x7fffffffaf14, maxevents=1024, timeout=-1) at ../sysdeps/unix/sysv/linux/epoll_wait.c:30
  2    Thread 0x7ffff79a8700 (LWP 339343) "main" futex_abstimed_wait_cancelable ( ) at ../sysdeps/nptl/futex-internal.h:320
  3    Thread 0x7ffff71a7700 (LWP 339344) "main" futex_abstimed_wait_cancelable ( ) at ../sysdeps/nptl/futex-internal.h:320
  4    Thread 0x7ffff69a6700 (LWP 339345) "main" futex_abstimed_wait_cancelable ( ) at ../sysdeps/nptl/futex-internal.h:320
  5    Thread 0x7ffff61a5700 (LWP 339346) "main" futex_abstimed_wait_cancelable ( ) at ../sysdeps/nptl/futex-internal.h:320

5.3 网络数据收发

5.3.1 数据接收

断点 recv

Thread 4 "main" hit Breakpoint 4, __libc_recv (fd=5, buf=0x7ffff69a5600, len=2048, flags=0) at ../sysdeps/unix/sysv/linux/recv.c:24
24      ../sysdeps/unix/sysv/linux/recv.c: No such file or directory.
(gdb) bt
#0  __libc_recv (fd=5, buf=0x7ffff69a5600, len=2048, flags=0) at ../sysdeps/unix/sysv/linux/recv.c:24
#1  0x000055555555df4a in HandleRecv::process (this=0x55555558bb90) at event/myevent.cpp:38
#2  0x000055555555c114 in ThreadPool::run (this=0x55555558b2b0) at threadpool/threadpool.cpp:114
#3  0x000055555555bc5c in ThreadPool::worker (arg=0x55555558b2b0) at threadpool/threadpool.cpp:77
#4  0x00007ffff7f8a609 in start_thread (arg=<optimized out>) at pthread_create.c:477
#5  0x00007ffff7c1c353 in clone () at ../sysdeps/unix/sysv/linux/x86_64/clone.S:95

很明显,这个项目是在线程池的线程里读取socket 数据并解析解析处理。

5.3.2 数据发送

断点 send

说明数据的响应也是在线程池的线程进行的。

6 拓展建议

建议将本项目的功能一直到muduo库进一步测试,对比分析性能的差异。

也建议和nginx做更多测试对比。

这样可以不断优化项目的性能,面试的时候也能有更多可以讲解的内容。