- 在线时间
- 0 小时
- 最后登录
- 2007-9-23
- 注册时间
- 2004-9-10
- 听众数
- 3
- 收听数
- 0
- 能力
- 0 分
- 体力
- 9975 点
- 威望
- 7 点
- 阅读权限
- 150
- 积分
- 4048
- 相册
- 0
- 日志
- 0
- 记录
- 0
- 帖子
- 1893
- 主题
- 823
- 精华
- 2
- 分享
- 0
- 好友
- 0

我的地盘我做主
该用户从未签到
 |
< >最近在写一个程序用到了多线程,所以对CB下的多线程有一定的学习。 t o M+ e6 _: i+ e$ S
现在把自己的一些心得讲一下。水平有限,写的很粗略,请大家见谅。</P>
. n* u8 v/ g' _, J$ r" b. l- D< >CB相对于VC来说,在CB下写多线程程序是很简单的。不仅是VCL中有TThread这个类。封装了那些关于多线程的WINDOW API。我觉得更方便的是他提供了7 g3 a1 b" v! i8 a6 W0 g8 |
直接访问主VCL线程中对象的能力。可以很容易的和主线程中的窗体,控件
, H# C( a4 T6 g; w7 h0 v2 P打交道。和单线程的方式没有太多区别。只是在有多个线程都要访问主线程
( [$ x8 t% N( x4 r中的对象(比如访问同一个窗体上的StringGrid).只要用Thread的Synchronize方法来调用那段访问主VCL线程的代码(具体请看帮助),我们就不用担心访问冲突的问题了。而且对于多线程的同步和互斥,CB也对WINDOW 编程中那些机制进行了封装。比如对临界区CriticalSection封装为TCriticalSection.事件Event封装为TEvent.这些类相当简单好用。
% g5 ?, ^3 b" H. x8 t下面就是我觉得比较重要的几点,供大家参考.</P>$ R3 W# w& C9 w. n# X
< >: t/ T8 t, M9 G$ K& M/ N3 n
1。TThread的WaitFor方法。是等待一个线程返回。其返回值在这个线程里可以任意设定。以便在该线程返回的时候让调用他的线程知道他的运行情况。
8 A! z+ n+ R9 N2 c( }
; I; z+ B8 {4 u在TThread的 OnTerminate事件中做线程的清除工作。他不是线程运行的一部分。
# y) O6 Z( L% X; J( H+ |而是主VCL线程的一部分。所以在其中不能访问Thread的局部变量(如 int __thread i) ; M4 M! Y! V U) d5 B, D4 E% H
你可以把清楚代码写在这里,不用管现在在EXCUTE()方法执行到了哪个地方。
) c2 I/ ~ e9 Q; y这么看起来有点类似于C++里的 finally 块的作用。 % ?! G4 A2 C' F: a4 ]8 m
: X a i3 g/ p! i, Q
2。TEvent很重要。实现线程的同步。WaitFor(int Timeout)功能类似于
) I. U& x9 y! ]' ZWINDOW API WaitforSingleObject().返回值包括: 1 I% g/ T" K8 s' P) ~
其中参数Timeout可以设为INFINITE表示永久等待,但这样,程序很容易死在这里。
# u! Q& K) _8 w ! r0 O W+ W+ Q2 r4 `% o# X1 i# H
wrSignaled 该事件发生(成功返回).
" r* y# g) n* b# B! }7 B4 p# D5 EwrTimeout 等待超时. * {/ e3 u: Q" q0 s9 w
wrAbandoned 在该事件的超时期限到达前,该事件对象已经被毁灭了。.
! d5 N1 y0 Y) S# n5 F; hwrError 在等待过程中有异常产生,要知道具体产生的错误要查看 TEvent的LastError
2 X# {! ]& ~, L& k( K% V7 [$ c属性。
1 S+ C3 L5 J; S# a/ [3 h6 R 5 R& o/ W! `$ } ~( v$ X
3? TCriticalSection
# m- z8 k6 K: p这个相当于WIN32编程中的临界区。
4 @$ S: y& ~1 s: ~ `7 K在多线程编程中,多个线程需要访问同一个公用变量的时候。 7 U3 ~" |& M" P) f# C5 b
7 s: Q- |7 \3 j. W
来保证访问的正确性。对公用变量访问的代码写在Enter();和Leave()之间。
) Y. {, J# g3 ?+ {% e比如有个公用变量 Count;
7 d! ?9 z" q$ {* @3 W以下代码 : : } N8 S& e$ q
TCriticalSection * pSection=new TCriticalSection();
) d( b* q+ Z/ v# m% k pSection->Enter(); . ^" [% m+ o! |3 R! l
Count++; 4 l {) t& m" O- [3 K
pSection->Leave(); 2 `' Z: ?0 n$ Y+ b
delete p;
! l5 w Y- a$ w0 e+ L0 I3 B I
, e& F6 e1 {7 ]. s6 Z' pEnter()方法进入临界区,对其中的公用变量加锁。 / w' q$ E4 I5 v! X/ Q
Leave()方法离开临界区,对其中的公用变量解锁。
0 j" V/ z' S; N4 O$ [
; f# v8 E) u' ?, ~! L$ C
) P- w7 D1 z9 m, Q* U4.TMultiReadExclusiveWriteSynchronizer % U ~: h# m+ j* D
用来处理类似于多个生产者和多个消费者的问题。这里的消费者是指
. ~% m8 l) n! q* Q3 h对公用变量进行读操作的线程。
2 Z+ f& [8 S7 m5 |, b* z) [( k生产者是对公用变量进行写操作的线程。</P>6 t4 i8 g4 H2 Z4 b/ l
< >四个方法。
3 y( x* m0 ?! U @. s BeginRead+ ]1 f- j1 {" }0 `. P% B& o
EndRead6 ]4 |5 a) j# I+ {$ Z* N6 s
这两个方法用于消费者。9 i& P2 u- w( p# }4 @
BeginWrite
- M9 n" v9 h+ a4 p1 ^ EndWrite: ]' J- Z1 B0 _; O! _5 Q
这两个方法用于生产者。</P>
# F" C2 @7 J- `" z: y6 M0 b! Z< >使用的时候就是要把这个TMutiReadExclusiveWriteSynchronizer 定义一个全局变量。5 t! G3 X( X! B2 U. ~
然后在其他线程中访问他。</P>
% k2 H) j( a+ G- {8 d0 y+ o6 J+ R% @
l e" @0 L+ Q7 x- @< >, v& ]+ O% @% d8 r9 O8 Z; |
</P> |
zan
|