如何用拼音码查询DBGRIDEH中的记录(100分)

  • 主题发起人 主题发起人 CHINESEWHOAMI
  • 开始时间 开始时间
C

CHINESEWHOAMI

Unregistered / Unconfirmed
GUEST, unregistred user!
现有一控件EDIT和一控件DBGRIDEH表(有几十万条记录),怎样在EDIT控件中录入拼单码,DBGRIDEH中把相对应的汉字过滤出来.
如录入BZ后,DBGRIDEH中只显示'本子'开头的记录.

谢谢!
 
请高手指教!谢谢!
 
类似的想法我也做过,我是查通迅录的,在服务端用SQL函数支持,先求得中文的拼音,再与输入的查询关键字比较,但这样速度不怎滴。而且也实现不了像你说的这种情况。所以后来改变了一下想法,对需要查询的字段,增加一个保存拼音首码的字段,在输入中文的时候,即刻求得拼音首码,查询时,根据输入的内容,判断一下是查原中文还是拼音首码。这样做后,效率要比在查询时动态用函数取要快很多,同时也可以实现前台的过滤功能。
 
levi说的对,应该有一个拼音码字段,这样查询方便、速度快。
 
找个拼音函数,然后查询一下。
 
如按levi说得话,那几十万条都加拼音码字段,那不是工作量很大?
我这边有个例子,只是是LISTBOX,改来改去不能用到DBGRIDEH中,还请高手指教一下
例子:
摘自大富翁论坛!

下面的内容来自Hubdog的 菜花宝典
---- 在日常工作和生活中我们经常使用电子记事本查找个人通讯录信息,或在单位的应用程序中查询客户档案或业务资料,这个过程中往往需要输入大量的汉字信息,对于熟悉计算机的人这已经是一件头疼的事,那些不太熟悉计算机或根本不懂汉字输入的用户简直就望而生畏。作为对数据检索技术的一种新的尝试,作者探索使用汉字拼音的首字符序列作为检索关键字,这样,用户不必使用汉字,只须简单地键入要查询信息的每个汉字的拼音首字符即可。比如你想查找关键字“中国人民银行”,你只需要输入“zgrmyh”。作者希望通过下面的例子,为广大计算机同行起一个抛砖引玉的作用,让我们开发的程序更加便捷、好用。

---- 原理很简单,找出汉字表中拼音首字符分别为“A”至“Z”的汉字内码范围,这样,对于要检索的汉字只需要检查它的内码位于哪一个首字符的范围内,就可以判断出它的拼音首字符。

---- 程序更简单,包括3个控件:一个列表存放着所有待检索的信息;一个列表用于存放检索后的信息;一个编辑框用于输入检索关键字(即拼音首字符序列)。详细如下:

---- 1.进入Delphi创建一个新工程:Project1

---- 2.在Form1上创建以下控件并填写属性:

控件类型 属性名称 属性值
Edit Name Search
ListBox Name SourceList
Items 输入一些字符串,如姓名等,用于提供检索数据
ListBox Name ResultList
 

---- 3.键入以下两个函数

// 获取指定汉字的拼音索引字母,如:“汉”的索引字母是“H”
function GetPYIndexChar( hzchar:string):char;
begin
case WORD(hzchar[1]) shl 8 + WORD(hzchar[2]) of
$B0A1..$B0C4 : result := 'A';
$B0C5..$B2C0 : result := 'B';
$B2C1..$B4ED : result := 'C';
$B4EE..$B6E9 : result := 'D';
$B6EA..$B7A1 : result := 'E';
$B7A2..$B8C0 : result := 'F';
$B8C1..$B9FD : result := 'G';
$B9FE..$BBF6 : result := 'H';
$BBF7..$BFA5 : result := 'J';
$BFA6..$C0AB : result := 'K';
$C0AC..$C2E7 : result := 'L';
$C2E8..$C4C2 : result := 'M';
$C4C3..$C5B5 : result := 'N';
$C5B6..$C5BD : result := 'O';
$C5BE..$C6D9 : result := 'P';
$C6DA..$C8BA : result := 'Q';
$C8BB..$C8F5 : result := 'R';
$C8F6..$CBF9 : result := 'S';
$CBFA..$CDD9 : result := 'T';
$CDDA..$CEF3 : result := 'W';
$CEF4..$D188 : result := 'X';
$D1B9..$D4D0 : result := 'Y';
$D4D1..$D7F9 : result := 'Z';
else
result := char(0);
end;
end;

// 在指定的字符串列表SourceStrs中检索符合拼音索引字符串
PYIndexStr的所有字符串,并返回。
function SearchByPYIndexStr
( SourceStrs:TStrings;
PYIndexStr:string):string;
label NotFound;
var
i, j :integer;
hzchar :string;
begin
for i:=0 to SourceStrs.Count-1 do
begin
for j:=1 to Length(PYIndexStr) do
begin
hzchar:=SourceStrs[2*j-1]
+ SourceStrs[2*j];
if (PYIndexStr[j]<>'?') and
(UpperCase(PYIndexStr[j]) <>
GetPYIndexChar(hzchar)) then goto NotFound;
end;
if result='' then result := SourceStrs
else result := result + Char
(13) + SourceStrs;
NotFound:
end;
end;

4.增加编辑框Search的OnChange事件:
procedure TForm1.SearchChange(Sender: TObject);
var ResultStr:string;
begin
ResultStr:='';
ResultList.Items.Text := SearchByPYIndexStr
(Sourcelist.Items, Search.Text);
end;
 
上面讲的方法是可以实现的,就是所谓的函数方法,这种方法的一个特点就是关键字不停变换的时候,值总是要不停的重新求过,当记录多的时候,会消耗大量时间。尤其是几十万条记录的表。 还有就是你要在DBGIRD中显示查找的结果,最常用的就是filter语句,因为这样做如果目标表已经下载到内存中的话,更改关键字,并不需要重新从服务器下载数据,如果关键字更改频繁的话,显示用FITER是较好的,可惜它不支持任何函数。无论是SQL的还是DELPHI的。

当然,求一个十几万的表的拼音首码,会增加时间与空间的开销,但相当于频繁的查询所需要等待的时间,你就会觉得,这种开销是很值得的,打个比方,我花一个小时的时间(完全可以在服务器空闲的时候做),然后能把查询的时间从3S提高到0.5S,这还不满足吗?

况且除了第一次建字库的时候需要先取得初值外,其余后面的增加与修改都可以做到触发器或者程序中处理(即后续记录的首码可以分散到每一条记录的增、删、改时处理,不再需要对整表求首码),对用户而言,单条记录的增、改并不需要等待多长时间。
 
后退
顶部