Subject: 在delphi中使用go语言的defer方法 [Print This Page]
Author:
zzzl Time: 2010-12-2 09:45 Subject: 在delphi中使用go语言的defer方法
在实现函数时,如果中间的步骤出错,需要释放资源并退出函数,这些工作很繁杂,容易出错。 go语言的作者对过去十年软件开发的经历感到失望,针对这个问题,他带来了defer方法,它能让不管在函数内的哪个地方exit,都确保你有机会清扫干净。
program demo_defer;
{$APPTYPE CONSOLE}
uses
SysUtils,
coroutineUnit; //还是用这个单元。。。还是在附件里
begin
TProc(procedure() //这个函数演示将一个文件的内容,拷到另一个文件里
var
f1, f2: Integer; //两个文件指针,f1的内容要拷到f2里
begin
f1:=FileOpen('f1', fmOpenRead); //打开f1文件
defer(procedure() //defer函数将参数函数保存起来,在它所属的函数退出时再调用。
begin
FileClose(f1);
Writeln('f1被关闭');
end);
f2:=FileOpen('f2', fmOpenWrite); //再打开f2文件
if f2=-1 then begin
Writeln('f2打开失败');
Exit; //果断退出,不必考虑f1的状态
end;
//copyContent(f1, f2); //开始copy(假设有这个拷贝函数存在)
FileClose(f2);
end)();
Readln;
end.
以往在处理这种情形时,需要判断f2打开是否失败,如果失败的话需要将f1关闭再退出,如果这个函数很复杂,有可能会忘记关闭,而用这个方法,确保不管你写多少个exit,关闭f1的代码都会被执行到。
实际输出的结果是 :
f2打开失败
f1被关闭
Attachment:
coroutineUnit.zip (2010-12-2 09:45, 1 K) / Download count 493
http://bbs.cnpack.org/attachment.php?aid=790
Author:
zzzl Time: 2010-12-2 13:47
周兄,这个是不是应该发到源啊,还可以再到那里贴吗。。
Author:
Passion Time: 2010-12-6 20:49
把此dpr的例子也作为附件传上来吧。我发到网站的文档中心去。
Author:
zzzl Time: 2010-12-8 11:46
好的
Attachment:
demo_defer.rar (2010-12-8 11:46, 379 bytes) / Download count 521
http://bbs.cnpack.org/attachment.php?aid=793
Author:
jAmEs_ Time: 2010-12-8 18:05
是不是D2010以上才支持?
Author:
khzide Time: 2011-1-19 12:06
你的这个coroutine单元只能是个演示,用在工程中还有不少问题.有待完善.你为什么不写完善了呢.又多不了多少代码.
Author:
zzzl Time: 2011-1-19 15:16
确实只是个演示,我在后面的贴子中,甚至把那个单元改名成了concept,并且尽量让代码短小,这样别人就可以很快的弄懂它,可以自己根据需要修改,加入到自己的项目中
Author:
khzide Time: 2011-1-19 23:02
有一处情况建议加上,就是go语句的匿名线程不知道什么时候结束。 使用go语句应该意为着这个工作会比较耗时。 而很多时候需要检查事情是不是已经做完了。建议能过一个ID标识出这个匿名线程。并可在其它时候再次检索它。
Author:
khzide Time: 2011-1-19 23:12
我在以下情景中用到go语句,有个问题:timer事件中 去访问一个可能被其它地方正在独占的对象。 当发现对象被独占时就等待。但不能阻塞timer事件。否则timer事件会越积越多。我希望下次timer事件来了后,检索一下上一个事件中go()的操作是否已经完成了。完成了就重新执行。还在执行的就跳过。这种情况用go怎么优雅的解决。楼主可以考虑一下。
Author:
zzzl Time: 2011-1-20 14:59
原帖由 khzide 于 2011-1-19 23:02 发表
有一处情况建议加上,就是go语句的匿名线程不知道什么时候结束。 使用go语句应该意为着这个工作会比较耗时。 而很多时候需要检查事情是不是已经做完了。建议能过一个ID标识出这个匿名线程。并可在其它时候再次检索它。 ...
你可以把go()改成函数,让它返回一个 TThread 对象,然后外部就可以通过它的 Finished 属性判断它是否已结束了。
另外我建议,外部代码,应该对线程的生存期做到无关化,就是不要让上层的业务逻辑和线程是否结束耦合起来,而是和线程的工作结果相关联,这也是我没让go()有返回值的原因。
Author:
zzzl Time: 2011-1-20 15:37
原帖由 khzide 于 2011-1-19 23:12 发表
我在以下情景中用到go语句,有个问题:timer事件中 去访问一个可能被其它地方正在独占的对象。 当发现对象被独占时就等待。但不能阻塞timer事件。否则timer事件会越积越多。我希望下次timer事件来了后,检索一下上一个事件 ...
可能没有理解好题意,这样行吗?
var done: Boolean=True;
procedure TForm1.Timer1Timer(Sender: TObject);
begin
if done then begin
done:=False;
go(procedure()
begin
ftp.download('...');
done:=True;
end);
end;
end;
Author:
zzzl Time: 2011-1-20 16:00
如果只是单纯的在多线程中独占的使用对象,可以用system.tmonitor
Author:
khzide Time: 2011-1-20 22:37
我的本意是不想在外部保存这个thread.或其它变量,让go做更多的事情,比如说通过ID访问go函数得到以前创建的线程。但后来发现没有必要。就像现在这样就可以了。保存了它的简洁性。 功能越多,限制也就越多。 我有个问题,这个thread没有用freeonterminate属性,线程结束后。这个匿名线程类怎么析构阿。 我用的delphi2010里没有匿名线程。我只能用helper类扩展了它。手动实现创建一个线程返回。不知道楼主这个环境是什么样的。是不是也存在这个问题。还是说不理它。让它执行完毕后一直占着一块内存。谢谢
Author:
zzzl Time: 2011-1-21 08:58
其实XE下的匿名线程在构造函数中是有freeonterminate:=true;的,如果你自己实现它的话,也要加上这句
Author:
roubyehya Time: 2011-3-2 11:11
找了好久,终于找到了,谢谢
Author:
juejiang321 Time: 2011-6-22 16:20
学习了
Welcome to CnPack Forum (http://bbs.cnpack.org/) |
Powered by Discuz! 5.0.0 |