

什么是QtWebApp
QtWepApp是一个C++中的HTTP服务器库,其灵感来自Java Servlet。
「QtWebApp包含以下组件:」
- HTTP 1.1服务器(HTTP 1.1 Server)
- 模板引擎(Template Engine)
- 文件记录器(File Logger)
- Windows服务安装程序(Windows Service Installer)
- 演示应用程序(Demo Applications)
HTTP服务器在并发线程中处理传入请求。它支持持久连接、HTTPS、会话cookie和文件上传。
其中包括一个简单的支持多种语言的模板引擎,它用运行时值填充文本文件中的占位符。模板引擎还支持条件输出和循环。其他更大的模板引擎,如ClearSilver,也可以用作替代方案。
记录器插入Qt,并将日志消息从qDebug(…)重定向到qFatal(…)到文件,同时还添加了时间戳、线程ID、会话ID等附加属性。对记录器配置文件的更改将自动变为活动状态,而无需重新启动程序。
QtService组件使您能够将应用程序设置为Windows服务。
大约2MB的小内存需求使web服务器有资格用于嵌入式系统。但对于更大的网络服务来说,它也足够强大。
「有关如何使用库的教程,请参阅:」 http://stefanfrings.de/qtwebapp/tutorial/index.html
QtWebApp工程
QtWebApp工程包含了库代码和实例代码,「QtWebApp库下载链接:」 http://stefanfrings.de/qtwebapp/QtWebApp.zip,解压之后的工程目录如下图:

RTools增加WebServer
移植流程


- host和post:代表web服务器的IP地址和端口。公用Web服务器使用端口80,而内部Web服务器通常在端口8080上侦听。
- minThreads:代表始终保持运行的线程数量,用来确保一段时间不活动后的良好响应时间。
- maxThreads:QtWebApp可以同时处理多个http请求,该参数指定并发工作线程的最大数量。其值要根据机器性能而定(可以利用负载生成器等工具来确定)。
- cleanupInterval:Web服务器始终以空线程池开头,当HTTP请求进入时,将根据需要创建线程。空闲线程由计时器缓慢关闭。每隔一个cleanupInterval时间间隔(以毫秒为单位),服务器都将关闭一个空闲线程。
- readTimeout:设置通过打开大量连接而不使用它们,来保护服务器免受简单的拒绝服务攻击。静默连接将在设定的毫秒数后被关闭。通常情况下,是由Web浏览器来关闭连接。
- maxRequestSize:保护服务器免受非常多的HTTP请求而导致内存过载的影响。此值适用于常规请求。
- maxMultiPartSize:适用于网络浏览器将文件上传到服务器时发生的大部分请求。如果要接受10 MB的文件,由于HTTP协议开销,必须将此值设置得更大一些。
开发WebServer功能
- httpserver.cpp内容:构造函数设置指定服务目录;service方法实现将文件内容传输给客户端,如果文件不存在则返回404错误.
#include "httpserver.h"
#include "QDir"
HttpServer::HttpServer(QObject* parent)
: HttpRequestHandler(parent)
{
Q_UNUSED(parent)
}
HttpServer::HttpServer(QString path)
{
basePath = path;
}
void HttpServer::service(HttpRequest &request, HttpResponse &response)
{
QFile file(basePath + request.getPath());
if(file.open(QFile::ReadOnly))
{
response.setHeader("Content-Type", "application/octet-stream");
while (!file.atEnd() && !file.error())
{
QByteArray buffer=file.readAll();
response.write(buffer);
}
}
else
{
response.setStatus(404, "File not found");
}
}
- httpserver.h内容:
#ifndef HTTPSERVER_H
#define HTTPSERVER_H
#include "httprequesthandler.h"
using namespace stefanfrings;
class HttpServer : public HttpRequestHandler
{
Q_OBJECT
public:
HttpServer(QObject* parent=nullptr);
HttpServer(QString path);
void service(HttpRequest& request, HttpResponse& response);
private:
QString basePath;
};
#endif // HTTPSERVER_H
- 在构造函数中,获取ini文件listener组的配置。
WebServerTool::WebServerTool(QWidget *parent) :
QMainWindow(parent),
ui(new Ui::WebServerTool)
{
ui->setupUi(this);
listenerSettings = new QSettings(":/HttpServer/httpServer.ini", QSettings::IniFormat, nullptr);
listenerSettings->beginGroup("listener");
}
void WebServerTool::httpStartListener()
{
if(httpListener != nullptr)
{
httpListener->close();
delete httpListener;
httpListener = nullptr;
}
if(ui->lineEditHttpPath->text().isEmpty())
{
ui->textBrowserHttpLog->append("请选择http服务目录");
return;
}
listenerSettings->setValue("host", ui->comboBoxHttpIp->currentText());
listenerSettings->setValue("port", ui->lineEditHttpPort->text());
httpServer = new HttpServer(ui->lineEditHttpPath->text());
httpListener = new HttpListener(listenerSettings, httpServer, nullptr);
if(httpListener == nullptr)
{
ui->textBrowserHttpLog->append("Http 启动监听失败");
}
else
{
ui->textBrowserHttpLog->append("Http 启动监听成功");
}
ui->buttonHttpStart->setText("停止(Stop)");
.......
}
void WebServerTool::httpStopListener()
{
if(httpListener != nullptr)
{
httpListener->close();
delete httpListener;
httpListener = nullptr;
}
ui->buttonHttpStart->setText("启动(Start)");
....
}
验证WebServer功能


wget http://IP:PORT/RToolTest.txt
- 当服务目录下存在我们请求的文件,执行结果:

- 当服务目录下不存在我们请求的文件,执行结果:
