文档章节

使用 acl 的生成向导快速创建一个C++ WEB 服务器

郑树新
 郑树新
发布于 2014/09/03 12:11
字数 1649
阅读 31
收藏 0

      在文章《使用 acl 生成向导快速创建服务器程序》中介绍如何使用 acl 的服务器生成向导自动创建服务器程序的过程,文章《使用 acl_cpp 的 HttpServlet 类及服务器框架编写WEB服务器程序》介绍了编写 HTTP 服务器的大体过程,本节将介绍如何使用 acl 提供的生成向导快速创建一个 WEB 服务器的过程。在 acl 项目的根目录下编译完所有的程序后(运行 make all),然后进入 app/wizard 目录编译向导程序(运行 make,win32 下可以使用 vc2003/vc2008/vc2010 编译),运行向导程序的过程如下:

 

$./wizard  #运行后向导程序提示如下:
select one below:
m: master_service; d: db; h: http; q: exit

#其中 'm' 表示生成一般性的服务器程序,'h' 表示可以生成 HTTP 类服务器程序,所以此处选  'h',接着向导程序提示输入所生成的服务器程序的名称,如可输入:http_server
please input your program name: http_server

#接着提示 HTTP 应用的类型,目前仅支持自动生成 HTTP 服务,所以只能选择 's'
please choose one http application type: s
s: http servlet
>s

#接着向导程序提示是否需要添加 cookie 的支持,可以选择 'y'
Do you want add cookie? y[y/n]: y

#输入 cookie 名称
Please enter cookie name: mycookie
create http_server/http_servlet.cpp ok.
#最后提示所采用的服务器模板类型:'t' 表示采用多进程多线程模式,'p' 表示采用多进程模式
choose master_service type:
        t: for master_threads
        p: for master_proc
>t

#向导程序便会自动生成以下文件:源文件,工程文件等
create http_server/http_server.cf ok.
create http_server/Makefile.in ok
create http_server/main.cpp ok
create http_server/master_service.h ok
create http_server/master_service.cpp ok
create http_server/http_servlet.h ok
create master_threads ok!
create http_server/Makefile ok.
create http_server/valgrind.sh ok.
create http_server/http_server.sln ok.
create http_server/http_server.vcproj ok.
create http_server/http_server_vc2012.sln ok.
create http_server/http_server_vc2012.vcxproj ok.
create http_server/http_server_vc2012.vcxproj.filters ok.
create http_server/Makefile.in ok
create http_server/stdafx.h ok
create http_server/stdafx.cpp ok
create common_files ok!

       OK,一个简单的 HTTP 服务器程序就生成了,整个过程不超过 10 秒。由上面显示可以看到向导生成的文件有:

 

       1、源程序文件:

             main.cpp,stdafx.cpp,master_service.cpp,http_servlet.cpp,

             stdafx.h,master_service.h,http_servlet.h

       2、配置文件:http_server.cf

       3、工程文件:Linux 下为 Makefile.in, Makefile;在 windows 下为 VC2003 的工程文件(http_server.sln,http_server.vcproj),以及 VC2012 的工程文件(http_server_vc2012.sln,http_server_vc2012.vcxproj,http_server_vc2012.vcxproj.filters)

       4、LINUX 下用来检测内存泄露的工具脚本:valgrind.sh

 

       在 http_server 目录下直接运行 make 编译,便可生成 http_server 可执行程序。该程序既可以放在 acl_master 服务器框架下运行(生产环境中),又可以以命令行方式手工直接运行(调试环境下)。为使浏览器可以直接访问该 WEB 服务,先打开 main.cpp 文件,将服务器监听地址(const char* addr)由 “127.0.0.1:8888” 改为 ":8888",然后再 make 重新生成 http_server 程序,手工运行:http_server 提示如下:

$./http_server alone   # 其中的参数 alone 通知程序以手工方式运行,可参见 main.cpp 中的代码
listen on: :8888
event_limit: max fdsize: 100001
src/event/events_epoll_thr.c(599), event_epoll_alloc_thr, use thread events - epoll

       然后在浏览器输入该服务器地址:http://192.168.188.112:8888/,得到的结果如下:

该 XML 文件并未包含任何关联的样式信息。文档树显示如下。
---------------------------------------------------------------------
      <root>
             <sessions>
                    <session sid="xxxxxx"/>
             </sessions>
             <params>
                    <param name1="null"/>
                    <param name2="null"/>
             </params>
      </root>

 

        下面为向导程序生成的主要源程序:

        一)、master_service.cpp

         在该文件中,需要关注四个函数:thread_on_accept、thread_on_read、thread_on_timeout、thread_on_close。这四个函数的参数都是 acl::socket_stream* 网络连接流对象。它们被调用的时机为:

         a、不接收到客户端连接请求时,acl 服务器框架会自动建立与客户端的连接, 同时回调 master_service::thread_on_accept(acl::socket_stream*) 虚函数,应用在此处生成 HTTP 服务器连接对象,并将其与 acl::socket_stream* 网络流关联:

bool master_service::thread_on_accept(acl::socket_stream* conn) 
{
        logger("connect from %s, fd: %d", conn->get_peer(true),
                conn->sock_handle());
        conn->set_rw_timeout(5);   // 设置客户端连接读超时时间

        http_servlet* servlet = new http_servlet();   // 创建 HTTP 服务对象
        conn->set_ctx(servlet); // 将该 HTTP 服务对象置入连接流中

        return true;
}

         b、当客户端发来 HTTP 请求时,acl 服务器框架将回调 master_service::thread_on_read(acl::socket_stream*) 虚函数,代码如下:

bool master_service::thread_on_read(acl::socket_stream* conn)
{
        http_servlet* servlet = (http_servlet*) conn->get_ctx();  // 取出在 thread_on_accept 中设置的 HTTP 服务对象
        if (servlet == NULL)
                logger_fatal("servlet null!");

        // 调用 HTTP 服务过程,如果该函数返回 true 则表示需要与浏览器保持长连接,否则则告诉服务器框架需要关闭该客户端连接
        return servlet->doRun("127.0.0.1:11211", conn);
}

        c、如果 HTTP 客户端长时间不发送 HTTP 请求数据,则虚方法:master_service::thread_on_timeout(acl::socket_stream*) 将会由服务器框架调用,该函数若返回 false 则通知服务器框架需要关闭该超时的客户端连接,否则表示继续等待客户端发来请求(超时值在 thread_on_accept 中设置)。

        d、当客户端连接超时、客户端主动关闭连接或服务端主动关闭连接时,则虚方法 master_service::thread_on_close(acl::socket_stream*) 将由服务器框架调用,代码如下:

void master_service::thread_on_close(acl::socket_stream* conn)
{
        logger("disconnect from %s, fd: %d", conn->get_peer(),
                conn->sock_handle());

        // 取出在 thread_on_accept 中设置的 HTTP 服务对象,若非空则销毁之
        http_servlet* servlet = (http_servlet*) conn->get_ctx();
        if (servlet)
                delete servlet;
}

 

        以上四个虚函数非常简单易懂,但有一点需要注意:这四个虚函数的线程运行空间是不同的,thread_on_accept 方法是在服务进程的主线程空间运行的,而其它的三个虚方法则是在线程池中的任一子线程中运行的,且线程空间各不相同,即使对于 thread_on_read 方法,当同一客户端连接对象反复回调该方法时,其运行的线程空间也可能是不同的。

 

        在 thread_on_accept 中生成了一个 HTTP 服务类:http_servlet,该类的接口类似于 JAVA 的 HttpServlet 类,使用方法也类似。

        二)、http_servlet.cpp/http_servlet.h

        该类 (http_servlet) 继承于 acl::http_servlet 类,其需要重载基类的两个虚方法:doGet/doPost,关于如何使用该类,可以参考:《用C++实现类似于JAVA HttpServlet 的编程接口》,在此不再细讲。

 

         acl 库下载:http://sourceforge.net/projects/acl/

         svn:svn checkout svn://svn.code.sf.net/p/acl/code/trunk acl-code

         github:https://github.com/zhengshuxin/acl

         acl 的编译与使用:http://zsxxsz.iteye.com/blog/1506554

         qq 群:242722074

       

        文章参考:

         使用 acl::master_threads 类编写多进程多线程服务器程序

         用C++实现类似于JAVA HttpServlet 的编程接口

         acl 服务器模块的部署

 

© 著作权归作者所有

郑树新

郑树新

粉丝 104
博文 87
码字总数 161171
作品 2
昌平
程序员
私信 提问
使用 acl 生成向导快速创建服务器程序

在以前有关使用 acl 的技术文章(如:使用 acl::masterthreads 类编写多进程多线程服务器程序 ,用 acl::masteraio 类编写高并发非阻塞服务器程序,使用 acl::master_proc 类编写多进程服务器...

郑树新
2014/09/03
0
0
Openbiz Cubi 快速应用开发向导

Openbiz Cubi 开发向导 在Cubi快速入门章节中提供了一个关于创建新模块的简单的范例,本章节将带领应用开发人员更加深入广泛的了解Cubi内部是如何工作的。理解Cubi内部的工作原理将有助与您在...

吉贤
2012/04/10
0
0
acl http服务器响应缺失http头问题

@郑树新 你好,想跟你请教个问题: 我使用acl快速向导建立了http服务器,有的客户端请求的时候,返回缺失http的错误,不知到哪里出问题了。 看log文件,只打印: (pid=25098)(info): master_s...

布尔凯索
2016/01/27
118
1
跨平台网络通信与服务器框架 acl 3.2.0 发布

acl 3.2.0 版本发布了,acl 是 one advanced C/C++ library 的简称,主要包括网络通信库以及服务器框架库等功能,支持 Linux/Windows/Solaris/FreeBsd/MacOS 平台;整个 acl 项目主要包含三个...

郑树新
2016/08/27
1K
4
acl.3.2.2 发布, 跨平台网络通信与服务器框架

acl 3.2.2 版本发布了,acl 是 one advanced C/C++ library 的简称,主要包括网络通信库以及服务器框架库等功能,支持 Linux/Windows/Solaris/FreeBsd/MacOS 平台;整个 acl 项目主要包含四个...

郑树新
2016/11/01
1K
2

没有更多内容

加载失败,请刷新页面

加载更多

面试爱奇艺,竟然挂在第5轮……

今天给大家分享我曾经在爱奇艺的面试,过程还是比较有意思的,可以给大家一些参考 <br> 聊骚阶段 嗲妹妹:你好,我是爱奇艺的HR,我们正在招聘运维开发岗位,请问您最近有在看工作机会吗? ...

上海小胖
今天
5
0
Jenkins系列_插件安装及报错处理

进入Jenkins之后我们可以进行插件的安装,插件管理位于以下模块: 发现上面报了一堆错误,是因为插件的依赖没有安装好,那么这一节,就先把这些错误解决掉吧。解决完成后,也就基本会使用插件...

shzwork
今天
2
0
mysql mysql的所有查询语句和聚合函数(整理一下,忘记了可以随时看看)

查询所有字段 select * from 表名; 查询自定字段 select 字段名 from 表名; 查询指定数据 select * from 表名 where 条件; 带关键字IN的查询 select * from 表名 where 条件 [not] in(元素...

edison_kwok
昨天
9
0
解决多线程并行加载缓存问题(利用guava实现)

依赖 com.google.guava:guava:20.0 import com.google.common.cache.Cache;import com.google.common.cache.CacheBuilder;import java.util.concurrent.ExecutionException;import j......

暗中观察
昨天
4
0
利用VisualVM 内存查看

准备工作,建几个测试类。等下就是要查看这几个类里面的属性 package visualvm;public class MultiObject { private String str; private int i; MultiObject(String str...

冷基
昨天
3
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部