开源[gsoap PostgreSQL libpq高性能JSON服务器](https://github.com/kmblack1/gservice)
服务流程图
代码请参看gSOAP 2.8.14 User Guide 中的7.2.4 How to Create a Multi-Threaded Stand-Alone Service.
listen : 监听soap_accept方法,返回一个SOAP_SOCKET,然后交SOAP_SOCKET添加到队列,并发送一个信号通知process(处理线程)处理.
process(处理线程)启动后立即阻塞等待信号,收到信号后才执行相应的任务(soap_serve
),执行任务后继续阻塞等待信号.注意:soap_serve方法如果客户端提交的xml文档格式不正确或其它数据(恶意攻击)时会一直阻塞到下一个请求,导到服务器性能严重下降,客户端无法收到数据一直等待的情况,因此在初始化处理线程的soap成功后,应该立即设置处理线程soap的超时时间,单位为秒:
soap_thr[i] = soap_copy(&soap);//在这一行示例代码后一加下面三行代码
soap_thr[i] ->send_timeout = 1;
soap_thr[i] ->recv_timeout = 1;
soap_set_mode(soap_thr[i], SOAP_C_UTFSTRING); /*设置采用UTF-8字符编码*/
例如下列的一个请求就会导致soap_serve阻塞(ns2:login节点没有闭合)
因为mod_gsoap附加到web服务器的方式问题比较多,iis比较复杂, apache只支持C开发模式,所以gsoap做为一个单独的服务运行,通过代理调用服务.iis可用其它插件或自己编写一个代理类,apache直接修改httpd.conf配置代理.
gsoap内存管理
尽量采用soap_malloc(用法和malloc一样)方法分析内存, soap_malloc无需释放内存,在gsoap内部采用链表的方式管理分配的内存,在执行soap_destroy(tsoap); soap_end(tsoap)后会自动回收内存.
自动回收内存的前提是保证你自己编写的方法能正确运行完成,否则还是会导致内存泄露.
gsoap输出
为提升服务性能,减少数据传输量,建议所有输出都采用字符方式的xml(UTF-8), 不要采用结构或对象输出(输入可以),采用结构或对象输出优点是在客户端无需解析,自动生成相关对象和结构,但是会导致服务性能下降和传输的数量增大.例如:
int ns2__login(xsd__string username,xsd__string password,xsd__string &rsp);
rsp为字符串格式的xml,在客户端需要解析后方可使用.我自己定义的rsp有三种输出格式,所有节点名称都大写,所有属性名称都小写,节点之间无换行,客户端按下面的三种规则编写解析器即可.
第一种为服务器异常消息
第二种为正常输入
更多ROW.......................................
更多ROWS.......................................
第三种为字段描述信息
更多字段描述信息.......................................
FIELDS >
更多FIELDS.......................................
type,size, required均为数字,type可以自己定义,解析时按自定义规则转换数据即可,size为字段大小, required值一般为0和1,为0表示不是必填,否则为必填项.
字符转换为保证可移值性,可采用iconv转换,windows和linux均支持,windows需要自己下载LibIconv for Windows - GnuWin32库.只需要记住UTF-8单个字符最多使用4字节存储即可.转换时一次性分配 tcslen(szBuffer) * 4 + sizeof(TCHAR)大小的内存.