答应过要发几篇心得的,第二篇:在 case 语句中使用字符串 (50分)

B

beta

Unregistered / Unconfirmed
GUEST, unregistred user!
声明:本文乃 熊恒(beta) 原创,如要转载请保持文章完整。
在 case 语句中使用字符串 --- by 熊恒(beta)
我今天要介绍的是一个比较另类的方法。大家都知道,case 语句只能对顺序类型
(ordinal type)管用,那么我们先看一下顺序类型到底有那些呢:1)整型;2)字
符型;3)枚举型。
于是也就演化出三种在 case 语句里面使用字符串的方法(耐心看下去哦,最后一
种方法才是今天的重点:p):
假如我们的具体应用如下(理想的,但是错误的写法):
str := 'Chongqing';
case str of // 编译器将在这一行制止你 :-(
'Beijing': ShowMessage('First');
'Tianjing': ShowMessage('Second');
'Shanghai': ShowMessage('Third');
'Chongqing': ShowMessage('Forth');
else
ShowMessage('Other');
end;

法一:字符串转为整型
这应该是比较通行的方法了,主要是利用 TStringList。先把备选的字符串挨个
Add 进去,然后调用其 IndexOf 方法。该方法返回一个整数,表示待找字符串出
现在列表中的位置:
//var strList: TStringList;
strList := TStringList.Create;
strList.Add('Beijing');
strList.Add('Tianjing');
strList.Add('Shanghai');
strList.Add('Chongqing');
str := 'Chongqing';
case strList.IndexOf(str) of
0: ShowMessage('First');
1: ShowMessage('Second');
2: ShowMessage('Third');
3: ShowMessage('Forth');
else
ShowMessage('Other');
end;
strList.Free;
可以看到,这是比较麻烦的方法了,不过你可得记住这个方法,因为后面的讨论
将会讲到,这是最通用的方法。
另:当然,把字符串转化为整型后用于 case 不只这一种方法。另一种是通过把
备选项全部转化为等长度后合并为一个字符串,然后用 Pos 函数返回某字符串的
位置。在此不再累述,请查阅《程序员》杂志(具体哪一期不记得了:-()。
法二:字符串转为字符型
这应该是最简单的一种方法,不过局限性比较大。如果你的备选项的第 N 个字
符(N 应为常数)都互不相同,那么你赚到了。这样,就可以通过取出这个字符,
来唯一标识你的字符串:
str := 'Chongqing';
case str[1] of // 第一个字符都不同,所以取出第一个进行比较
'B': ShowMessage('First');
'T': ShowMessage('Second');
'S': ShowMessage('Third');
'C': ShowMessage('Forth');
else
ShowMessage('Other');
end;

不过如果你的备选项没有这样的特性,那么你就无缘使用这种方法了。
法三:字符串转为枚举型
这种方法主要用到 RTTI 的特性。该方法的主要思路是,先把所有的备选项声名
为一个枚举类型的值,那么我们只要把要找的字符串也转换为枚举型,那么就可以
用 case 语句了。那么通过什么办法把一个字符串转化为枚举型呢?往下看:
// uses TypInfo;
// 记得引用这个单元
// type TMyStrSel = (Beijing, Tianjing, Shanghai, Chongqing);
// 注意,上面这个定义不能放在某个函数内部哦,那样的话,它就没有运行类信息了
// var strSel: TMyStrSel;
str := 'Chongqing';
strSel := TMyStrSel(GetEnumValue(TypeInfo(TMyStrSel), str));
case strSel of
Beijing: ShowMessage('First');
Tianjing: ShowMessage('Second');
Shanghai: ShowMessage('Third');
Chongqing: ShowMessage('Forth');
else
ShowMessage('Other');
end;

稍做解释:GetEnumValue 函数返回一个字符串对应的枚举型的值在某枚举类型
中的位置(要知道枚举类型是有顺序的,要不怎么叫顺序类型呢:)),然后通过
一个强制类型转换将这个值转化为枚举型。于是就实现了把字符串转化为枚举型的
操作。
可以看到,在 case 语句这一段,我们的使用和理想中的使用方法几乎是一样的!
毕竟枚举类型可以做到见名知意嘛。的确比用前两种方法看起来直观得多。
当然,这并不是万能的方法,如果你的备选项有一个不符合 Delphi 的变量名命
名法则(如'AK-47'或中文等),则不能将其声明为一个枚举型,于是就不能使用这
个方法。如果是这样你就只能使用前两种方法了。极端的情况下,至少你还有第一
种方法可以使用 :) 但在不少的场合,这种方法还是适用的。

 
B

beta

Unregistered / Unconfirmed
GUEST, unregistred user!
欢迎大家指正或补充!

 
B

barton

Unregistered / Unconfirmed
GUEST, unregistred user!
呵呵,不新鲜.我一直用数组来处理.
 
F

forss

Unregistered / Unconfirmed
GUEST, unregistred user!
暴好。。我喜欢。:)
以前也有找过。不过太麻烦的,就不想用了。你的比较简单。:))
 
B

beta

Unregistered / Unconfirmed
GUEST, unregistred user!
barton 大哥,多谢提醒,用数组到不失为一个好方法:)
 
L

lql0459

Unregistered / Unconfirmed
GUEST, unregistred user!
行是行,不过有一个问题:超过26个项目怎么办?(虽然这很少)
 
B

beta

Unregistered / Unconfirmed
GUEST, unregistred user!
to lql0459: 所以我强调了方法一一定要记住:)
 
C

coolbaby

Unregistered / Unconfirmed
GUEST, unregistred user!
很不错
:)
 

Unregistered / Unconfirmed
GUEST, unregistred user!
呵呵,我用的是另一个方法,
var
tempstr:string;
begin
tempstr:='test1,test2,test3,test4,.......' 把定义的字符串都弄成一个字串
case pos(str,tempstr) of 查找子字符串在tempstr中的位置
case 1: test1;
case N
end

end;

呵呵,这个方法有缺点,就是要知道每一个子字符串的位置,不过我觉得这样还是可以接受的
不需要创建别的对象,不需要user其它类,,
 
X

xianjun

Unregistered / Unconfirmed
GUEST, unregistred user!
上面提到的方法都用过
没有最好最坏之说,要看在什么情况下使用了。
另:
枫: 我一般使用这种方法的时候会这样用:
const
tempstr ='test1,test2,test3,test4,.......';
begin
case pos(str + ',', tempstr) of //如果","号是分隔符的话,我一般用“、”[:)]
1: test1;
...
end;
end;
这样 如果str是“test1,”的时候就不会误判了。
 

Unregistered / Unconfirmed
GUEST, unregistred user!
to xianjun
呵呵,谢谢了,受教………
 
P

pyzfl

Unregistered / Unconfirmed
GUEST, unregistred user!
学习,收藏!
 
B

beta

Unregistered / Unconfirmed
GUEST, unregistred user!
其实 枫 所说的方法,和我的第一种方法的“另”部分是差不多的,
不过你的方法是自己计算位置,比较麻烦:(
改进:
// const
// Max_Sel_Len = 10;
// 最大备选项长度
strSel := 'Beijing Tianjing Shanghai Chongqing';
// 上述每个备选项补足 Max_Sel_Len 字节
str := 'Chongqing';
case Pos(str, strSel) div Max_Sel_Len of
0: ShowMessage('First');
1: ShowMessage('Second');
2: ShowMessage('Third');
3: ShowMessage('Forth');
else
ShowMessage('Other');
end;

这样,case 里面的选项 0~3 就不用自己计算了。
其实这也就是我在第一种方法的后面介绍的方法了:)
 

Unregistered / Unconfirmed
GUEST, unregistred user!
嗯,其实这缺点我是知道,所以我也提出来了。其实还有一个不好的地方是,每改变一个备选
项都要重新记算位置,不知道有什么好办法解决这个问题呢,
to beta,
不过要是预定长度的话,也不是一个很好的办法,
1、需要预先知道最大的备选长度,
2、需要补足每一个备选的长度。
不过可以解决,改变一个备选项都要重新计算位置的问题呢。
 
B

beta

Unregistered / Unconfirmed
GUEST, unregistred user!
所以还是用我前面说的方法比较好啊:)
 
M

mygod!

Unregistered / Unconfirmed
GUEST, unregistred user!
支持,不过尽量不用string类型的啦
 
B

beta

Unregistered / Unconfirmed
GUEST, unregistred user!
to mygod!:的确,不过有时候身不由己啊:)
 

小雨哥

Unregistered / Unconfirmed
GUEST, unregistred user!
补充方法三的内容:
TMyStrSel = (Beijing, Tianjing, Shanghai, Chongqing);
// 定义它
{以下是使用举例}
procedure TForm1.Button1Click(Sender: TObject);
var
StrToOrd:integer;
begin
StrToOrd:=Ord(Chongqing);
StrSelect(StrToOrd);
end;

procedure TForm1.StrSelect(StrSel: Integer);
begin
case StrSel of
ord(Beijing): ShowMessage('First');
ord(Tianjing):ShowMessage('Second');
ord(Shanghai): ShowMessage('Third');
ord(Chongqing): ShowMessage('Forth');
else
ShowMessage('Other');
end;
end;
这样就不需要 beta 老大所说的 TypInfo 单元和 GetEnumValue 函数了。
(up 并收藏)
 

白马小将

Unregistered / Unconfirmed
GUEST, unregistred user!
不错,继续
以前我老是自己写函数转换
case myfunction of
1...
2...
3...
 

Similar threads

S
回复
0
查看
947
SUNSTONE的Delphi笔记
S
S
回复
0
查看
766
SUNSTONE的Delphi笔记
S
顶部