KBMMW SampleService/SampleClient方式传输数据集
马上周末了,趁着下午这会儿回顾一下这几天对旧项目的升级过程,一些重要但不常用的东西记录下来是很有必要的。其中一个项目中对KBMMW的远程数据通讯方式做了改进,利用SampleService/SampleClient方式传输数据集,以增加对底层数据通讯的可控性。
服务端代码示例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
|
type TkbmMWSimpleService1 = class (TkbmMWSimpleService) private { Private declarations } protected { Protected declarations } function ProcessRequest( const Func: string ; const ClientIdent:TkbmMWClientIdentity; const Args: array of Variant):Variant; override; function PerformOpenSQL(ClientIdent:TkbmMWClientIdentity; const Args: array of Variant):Variant; virtual; public { Public declarations } {$IFNDEF CPP}class{$ENDIF} function GetFlags:TkbmMWServiceFlags; override; end ; //...省略 function TkbmMWSimpleService1 . ProcessRequest( const Func: string ; const ClientIdent:TkbmMWClientIdentity; const Args: array of Variant):Variant; var AFunc: string ; begin AFunc:=UpperCase(Func); if AFunc= 'OPENSQL' then Result:=PerformOpenSQL(ClientIdent,Args) else Result:= inherited ProcessRequest(Func,ClientIdent,Args); end ; function TkbmMWSimpleService1 . PerformOpenSQL(ClientIdent:TkbmMWClientIdentity; const Args: array of Variant):Variant; var sqlStr: string ; aQuery: TUniQuery; aconn: TkbmMWUNIDACConnection; memTable: TkbmMemTable; aStreamFormat: TkbmBinaryStreamFormat; begin Result := 0 ; sqlStr:=args[ 0 ]; aQuery := TUniQuery . Create( nil ); //uniquery,和两层的用法一样 aQuery . Options . QueryRecCount := True ; aconn := TkbmMWUNIDACConnection(DMunt . kbmMWUNIDACConnectionPool1 . GetBestConnection( True , 0 , nil , 10000 )); //取连接池中的连接 if aconn = nil then begin kbmMWRaiseServerException( '无可用的数据库连接' ); Exit; end ; aQuery . Connection := aconn . Database; aQuery . SQL . Text := sqlStr; if (mainform . PAL_mode_01 . Visible) then LogIOer . AddShow(ClientIdent . Username+ '执行SQL查询:' +aQuery . SQL . Text, 0 ); memTable := TkbmMemTable . Create( nil ); aStreamFormat := TkbmBinaryStreamFormat . Create( nil ); memTable . DefaultFormat := aStreamFormat; memTable . IndexFieldNames := '' ; memTable . SortFields := '' ; memTable . MasterSource := nil ; try try aQuery . Open; //查询 Result := aQuery . RecordCount; //返回记录条数 except on E:Exception do begin kbmMWRaiseServerException(E . Message+ ',异常语句:' +aQuery . SQL . Text); //抛异常到客户端 Exit; end ; end ; memTable . LoadFromDataSet(aQuery,[mtcpoStructure,mtcpoProperties]); //复制数据集进KbMemTable memTable . SaveToStreamViaFormat(ResultStream,aStreamFormat); //按照指定流格式存入结果流ResultStream finally aconn . UnlockConnection; aQuery . Free; memTable . Free; aStreamFormat . Free; end ; end ; |
客户端代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
|
function openSql(Sqlstr: string ; var Ds:TDataSet; var Rs: string ): integer ;stdcall; var args: array [ 0..1 ] of Variant; begin Result := 0 ; Rs := '执行成功' ; if assigned(kbmMWSimpleClient1) then begin try args[ 0 ]:= Sqlstr; //SQL语句 Result := kbmMWSimpleClient1 . Request( 'TkbmMWSimpleService1' , '' , 'openSql' ,args); //SimpleClient执行请求 kbmMemTable . EmptyTable; //清空内存表 kbmMemTable . LoadFromStreamViaFormat(kbmMWSimpleClient1 . ResultStream,aStreamFormat); //将结果流复制进内存表 ds := kbmMemTable; //返回dataset(kbmMemTable继承自Tdataset) except on E:Exception do begin Result := - 1 ; rs := errorInfo(E . Message); if FIsLog then Writelog(rs); end ; end ; end else begin Result := - 1 ; //未执行初始化操作 Rs := '远程数据通讯接口未执行初始化操作' end ; end ; |
核心代码就这些,相信用到的人能够看明白。同理,可以利用这种方式实现二进制文件流(如:图像等)的传输,不再赘述。
另外,有一个小问题折磨了我一下午,提醒大家一下,希望大家不要像我一样粗心:
有两个类:TkbmBinaryStreamFormat(kbmMemBinaryStreamFormat.pas)、TkbmMWBinaryStreamFormat(kbmMWBinaryStreamFormat.pas)很容易混淆(正确用法见上述代码),而且一旦混淆,会造成KbmMeMTable在流的处理过程中出错。
http://www.pfeng.org/archives/385
KBMMW SampleService/SampleClient方式传输数据集的更多相关文章
- Node.js初探之GET方式传输
Node.js初探之GET方式传输 例子:form用GET方法向后台传东西 html文件: <form action="http://localhost:8080/aaa" ...
- 流,用声明性的方式处理数据集 - 读《Java 8实战》
引入流 Stream API的代码 声明性 更简洁,更易读 可复合 更灵活 可并行 性能更好 流是什么? 它允许以声明方式处理数据集合 遍历数据集的高级迭代器 透明地并行处理 简短定义:从支持数据处理 ...
- C# Post方式传输报文,和处理响应
public string DoPost(string url, string data) { HttpWebRequest req = GetWebRequest(url, "POST&q ...
- Node.js初探之POST方式传输
小知识:POST比GET传输的数据量大很多 POST发数据--"分段" 实例: 准备一个form.html文件: <!DOCTYPE html> <html> ...
- ASP.NET WebServce项目下添加Http服务,支持Get,Post请求方式;传输格式json/xml
由于WEBServce老项目中需要增添新的接口,而且添加的接口不希望被其它项目以引用Servces方式使用. 那么得在现有Service项目中添加Http请求方式来实现系统间数据交互.只需要告知请求地 ...
- http使用formData方式传输文件请求
转载请注明出处: 项目中有遇到http使用formData请求传输文件,在此记录一下 1.依赖jar包: <dependency> <groupId>org.apache.ht ...
- 用socket方式传输Image和Sound文件
import java.io.File; import java.io.FileInputStream; import java.io.IOException; import java.io.Obje ...
- cache支持single/increment/increment4三种方式传输
1.cache bypass signle---data length 已知 increment ---data length 不知 用 last data address 结束数据传输 2.cac ...
- json和jsonp的传输方式
jsonp传输会解决跨域的问题 $.ajax({ async: false, /* url: "http://127.0.0.1:8080/2015020601/background/mea ...
随机推荐
- java常用系统包介绍
java.applet提供创建 applet 所必需的类和 applet 用来与其 applet 上下文通信的类.java.awt包含用于创建用户界面和绘制图形图像的所有类.java.awt.colo ...
- Sublime Text3快捷方式总结
Ctrl+P快速查找 Ctrl+D多行游标//同时选中多个地方进行编辑 Ctrl+F查找替换 Ctrl+H查找替换 Ctrl+G快速跳到某一行 Ctrl+shift+P命令模式 在命令模式下设置语法: ...
- ASP.NET之电子商务系统开发-4(二级分类)
一.前言 继上次的订单,这是第四篇.记录一下分类和筛选.这功能是最后做的,因为我完全不懂其原理.后来通过同学的指导(一位很有天赋的同学,比我牛逼一个层次,同样是高三.:D),终于也是完成了.在写这篇博 ...
- ASPxGridView后台获取edit、delete、选择框等按钮。
GridViewCommandColumn commandColumn = (GridViewCommandColumn)ASPxGridViewInstance.Columns["#&qu ...
- oracle 计算两个时间之间的月份差,相差几个星期,相差多少天
相差多少天: sysdate-to_date('1991-01-01','YYYY-MM-DD'))<7 and (sysdate-to_date('1991=01=01','YYYY-MM ...
- objective-C学习笔记(三)数据成员:属性与实例变量
类型成员 Type Member 结构体 struct 的成员很简单,只有变量. 类的成员就很多了: 数据成员 data member 描述对象(本讲重点) · 实例变量 instance vari ...
- 关于TableViewCell高度自适应问题的整理
TableViewCell高度自适应在网上有很多资料,我只想找出最最最简单的一种方法. 首先梳理一下思路.说到TableViewCell我们第一个想到的问题或许就是cell的复用问题. 1. [se ...
- C++静态局部对象
7.5局部对象 在C++语言中,对于每一个变量和对象,都有其各自的作用域和生存期,这两个概念一个是空间的,一个是时间的.对象的作用域指的是该变量的程序文本区,对象的生存期则是程序执行过程中对象存在的时 ...
- [C#编程参考]把图像转换为数组的两种实现
当一个程序和一个图片放在一起,无非有两种操作: 第一种,就是传输这个图片,在传输图片之前要首先把这个图片变成byte类型的数组.所以这时候我们用到的是图片的存储的数据,也就是图片属性中的大小.我们并不 ...
- [置顶] java 枚举
1. 什么是枚举?枚举就是用来存放一组固定的常量. 2. 枚举有什么作用?一些程序在运行时,它需要的数据不能是任意的,而必须是一定范围内的值:例如性别 男和女. public enum Gender ...