from:http://iunknown.javaeye.com/blog/78561
关键字: C++ libevent spserver http embedded web 服务器
看到这个题目,估计很多人会问:为什么要再实现一个 web 服务器?
这里有几个原因:
1.这是一个 web 服务器框架,不是一个完整的 web 服务器。也就是说 SPWebServer 提供的是一套 API 和类库,可以方便地集成到现有的应用程序中。可以称 SPWebServer 为 embedded web server 。
2.有些时候,我们需要的不是一个功能强大完整的 web 服务器(例如 apache ),我们只是需要一个能提供最基本的 http 功能的服务器框架。比如要用 C/C++ 实现 XML-RPC,JSON-RPC 服务器端,或者为 jabberd 实现一个 http bind 的时候。这些场景下,也可以使用 apache,但是使用 embedded web server 可以简化配置。
3.在如下的场合中,apache 显得不合适,而 embedded web server 却正好合适:在一个现有的应用程序中,用 http 接口来提供一些简单的控制和统计功能。
基于以上原因,也为了
1.丰富 SPServer 项目(请参考: SPServer : 一个基于 Half-Sync/Half-Async 模式的高并发 server 框架)的功能;
2.正好作为 SPServer 框架的一个比较好的示范例子;
因此基于 SPServer 框架实现了一个 web 服务器框架。
源代码下载:
http://spserver.googlecode.com/files/spserver-0.5.src.tar.gz
http://code.google.com/p/spserver/downloads/list
下面来看一个使用 SPWebServer 的简单例子。
data:image/s3,"s3://crabby-images/a6818/a68189c6e83a0785f35a469a9083474de3dc50cf" alt="clip_image001 clip_image001"
class SP_HttpEchoHandler : public SP_HttpHandler
{
public:
data:image/s3,"s3://crabby-images/65856/658561332d02c46188becf7034f175b8c503ebcc" alt="clip_image005 clip_image005"
SP_HttpEchoHandler()
{}
data:image/s3,"s3://crabby-images/3c7e9/3c7e9e80e0fb11f435e21d7bef516c3dc2ba679e" alt="clip_image005 clip_image005"
virtual ~SP_HttpEchoHandler()
{}
data:image/s3,"s3://crabby-images/ef26c/ef26cfe132cfc4c71e0259779df0aba7fc62da49" alt="clip_image005 clip_image005"
virtual void handle( SP_HttpRequest * request, SP_HttpResponse * response )
{
response->setStatusCode( 200 );
response->appendContent( "<html><head>"
"<title>Welcome to simple http</title>"
"</head><body>" );
data:image/s3,"s3://crabby-images/827d3/827d332b896d83331d4b9e25f22d761be5d09949" alt="clip_image005 clip_image005"
char buffer[ 512 ] =
{ 0 };
snprintf( buffer, sizeof( buffer ),
"<p>The requested URI is : %s.</p>", request->getURI() );
response->appendContent( buffer );
snprintf( buffer, sizeof( buffer ),
"<p>Client IP is : %s.</p>", request->getClientIP() );
response->appendContent( buffer );
data:image/s3,"s3://crabby-images/c504c/c504c9a6e659b44d0a333a46638ed65c31fbef05" alt="clip_image005 clip_image005"
for( int i = 0; i < request->getParamCount(); i++ )
{
snprintf( buffer, sizeof( buffer ),
"<p>Param - %s = %s<p>", request->getParamName( i ),
request->getParamValue( i ) );
response->appendContent( buffer );
}
data:image/s3,"s3://crabby-images/0dc6d/0dc6ddccf0766ecad9fe7bc5de2b1073e36d7e71" alt="clip_image005 clip_image005"
for( int i = 0; i < request->getHeaderCount(); i++ )
{
snprintf( buffer, sizeof( buffer ),
"<p>Header - %s: %s<p>", request->getHeaderName( i ),
request->getHeaderValue( i ) );
response->appendContent( buffer );
}
data:image/s3,"s3://crabby-images/e8ccb/e8ccb78142fc8ae790c5025a03d1ad74bf59ea4c" alt="clip_image005 clip_image005"
if( NULL != request->getContent() )
{
response->appendContent( "<p>" );
response->appendContent( request->getContent(),
request->getContentLength() );
response->appendContent( "</p>" );
}
response->appendContent( "</body></html>\n" );
}
};
data:image/s3,"s3://crabby-images/b670f/b670fd2ee0b58514cd77e0fa755ed59b49a0b1ce" alt="clip_image001 clip_image001"
class SP_HttpEchoHandlerFactory : public SP_HttpHandlerFactory
{
public:
data:image/s3,"s3://crabby-images/c3d49/c3d49e4c57cd8e58989fcdd0f4d85b7ba5cc76c0" alt="clip_image005 clip_image005"
SP_HttpEchoHandlerFactory()
{}
data:image/s3,"s3://crabby-images/30c9d/30c9d62026c44008560a8d11b9ec4da6e27138f0" alt="clip_image005 clip_image005"
virtual ~SP_HttpEchoHandlerFactory()
{}
data:image/s3,"s3://crabby-images/2e7be/2e7be55f7d7e5e718578f08a645efc42adfb3f16" alt="clip_image005 clip_image005"
virtual SP_HttpHandler * create() const
{
return new SP_HttpEchoHandler();
}
};
//---------------------------------------------------------
int main( int argc, char * argv[] )
data:image/s3,"s3://crabby-images/6a35e/6a35e84f1b03d032f35ff88f9a8998d3a01804d5" alt="clip_image001 clip_image001"
data:image/s3,"s3://crabby-images/3ea12/3ea1297e73bc6f55861513267570f799ebeab01d" alt="clip_image002 clip_image002"
{
int port = 8080;
SP_Server server( "", port,
new SP_HttpHandlerAdapterFactory( new SP_HttpEchoHandlerFactory() ) );
server.runForever();
return 0;
}
data:image/s3,"s3://crabby-images/2f6d1/2f6d1e8ccc31b194b4007aa65db39c7779fb6967" alt="clip_image009 clip_image009"
上面的代码演示的是一个 http echo server ,即把 client 发过来的请求信息显示在页面上。
在最简单的情况下,使用 SPWebServer 需要实现两个类:SP_HttpHandler 的子类 和 SP_HttpHandlerFactory 的子类。
SP_HttpHandler 的子类负责处理具体的 http 请求。
SP_HttpHandlerFactory 的子类协助 spserver 为每一个连接创建一个 SP_HttpHandler 子类实例。
在实现过程中,使用了 Adapter 模式,把 SP_HttpHandler 和 SP_HttpHandlerFactory 适配为 SP_Handler 和 SP_HandlerAdapterFactory。这两个 Adapter 都已经实现,不需要使用者来实现。