ODAC获取数据的效率比较高,在Web程序中希望能够更快获取第一页的数据时,可以有几种方式:
1、在数据库中进行分页处理;
2、获取所有数据,只是快速返回第一页数据。
第一种方案对应用服务器资源消耗最小,对数据库消耗略大,在客户需要对全数据进行灵活过滤、查找、统计时就有些不够用了,另外对耗时较大的SQL查询就不如第二种方案速度快,对数据库压力也要大些,并且需要编写程序来完成。在Delphi下我考虑使用第二种方案,尤其是在使用uniGUI+ODAC配合使用时。第二种方案对应于服务器内存压力略大,并且要求快速获取第一页数据,为此做了个实验:
1、单表中有24万条记录
2、使用一次性获取全部记录,并放在服务器内存中,由uniGUI的网格进行自动分页处理。
3、非一次性获取全部记录时,需要快速获取第一页数据,将TOraQuery组件的FetchRows设置与TUniDBGrid组件的WebOptions.PageSize一致(不是必须的,只是觉得这样对显示第一页更有效率些),再通过启动一个线程,在后台通过TOraQuery组件的FetchAll属性设为True来获取全部数据。TOraQuery的数据量发生变化后,在TUniDBGrid组件中滚动任意记录都会触发记录数与分页数的自动更新,所以不需要在获取全部数据后刷新网格。
代码如下:
procedure TMainForm.UniButton11Click(Sender: TObject);
var
d : DWORD;
begin
//计时
d := GetTickCount;
//每个数据包获取的记录数,建议与网格的每页数保持一致
UniMainModule.OraQuery7.FetchRows := UniDBGrid7.WebOptions.PageSize;
//是否一次性获取
UniMainModule.OraQuery7.FetchAll := UniCheckBox7.Checked;
//开启表,如果是非一次性获取,则中获得了第一个数据包的记录数
UniMainModule.OraQuery7.Open;
//花费时间
UniLabel17.Caption := Format('%d ms', [GetTickCount - d]);
//开启线程获取
if not UniCheckBox7.Checked then
TFetchThread.Create(UniMainModule.OraQuery7);
end;
UniCheckBox7.Checked决定了是否采用一次性获取的选项,实验显示:
1、一次性获取数据模式,显示第一页花费了3000ms左右时间,内存占用约180M,关闭数据集后内存减少为10M左右,说明内存释放非常干净。
2、非一次性获取数据模式,显示第一页花费了约20ms左右时间,后台读取数据没有影响前端数据的展示、滚动等操作,最终内存占用与关闭后释放同一次性获取模式。
后台获取数据的线程代码如下:
TFetchThread = class(TThread)
private
FDataSet : TOraQuery;
public
procedure Execute; override;
constructor Create(ADataSet : TOraQuery);
end;
{ TFetchThread } constructor TFetchThread.Create(ADataSet: TOraQuery);
begin
FDataSet := ADataSet;
FreeOnTerminate := True;
inherited Create;
end; procedure TFetchThread.Execute;
begin
inherited;
if Assigned(FDataSet) then
begin
FDataSet.FetchAll := True;
while not FDataSet.Fetched do
Sleep(); MainForm.Caption := 'refresh';
end;
end;
效果基本满意