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

我的地盘我做主
该用户从未签到
 |
< >最近在写一个程序用到了多线程,所以对CB下的多线程有一定的学习。
/ f) n, Y7 u! o/ K: q9 L现在把自己的一些心得讲一下。水平有限,写的很粗略,请大家见谅。</P>5 o& L* W) ^. {8 q3 z% ~
< >CB相对于VC来说,在CB下写多线程程序是很简单的。不仅是VCL中有TThread这个类。封装了那些关于多线程的WINDOW API。我觉得更方便的是他提供了
/ c. b3 l$ W8 e3 L直接访问主VCL线程中对象的能力。可以很容易的和主线程中的窗体,控件" W5 {8 O9 t9 D1 n) g
打交道。和单线程的方式没有太多区别。只是在有多个线程都要访问主线程
+ _' x; S# B2 C. B, X中的对象(比如访问同一个窗体上的StringGrid).只要用Thread的Synchronize方法来调用那段访问主VCL线程的代码(具体请看帮助),我们就不用担心访问冲突的问题了。而且对于多线程的同步和互斥,CB也对WINDOW 编程中那些机制进行了封装。比如对临界区CriticalSection封装为TCriticalSection.事件Event封装为TEvent.这些类相当简单好用。3 D' z+ E9 i4 I; D2 e5 L
下面就是我觉得比较重要的几点,供大家参考.</P># n% l* H6 z( w u- l2 E% }
< >
! e" g$ a3 z0 }1。TThread的WaitFor方法。是等待一个线程返回。其返回值在这个线程里可以任意设定。以便在该线程返回的时候让调用他的线程知道他的运行情况。 ) V$ x! [+ z9 ~( I' c4 w: R
3 t# Y6 s6 V" [! {& n: R! l2 |0 _# d5 Y
在TThread的 OnTerminate事件中做线程的清除工作。他不是线程运行的一部分。 2 t& `5 U2 M' x+ e- U! }
而是主VCL线程的一部分。所以在其中不能访问Thread的局部变量(如 int __thread i)
! W5 k+ R9 \" J, U" L. g你可以把清楚代码写在这里,不用管现在在EXCUTE()方法执行到了哪个地方。 , i3 T. B, k9 \* S* I
这么看起来有点类似于C++里的 finally 块的作用。 + R8 F4 o& i) F
: g' v. P2 x3 P C2。TEvent很重要。实现线程的同步。WaitFor(int Timeout)功能类似于
2 {$ x5 q' V" f0 t% _WINDOW API WaitforSingleObject().返回值包括:
6 Z8 \% r1 \/ A0 C# ^0 g其中参数Timeout可以设为INFINITE表示永久等待,但这样,程序很容易死在这里。
2 T- H `" O: i7 ^
" q; J' }+ [) O+ T( R0 zwrSignaled 该事件发生(成功返回). ! N4 D! } O* _( u" i
wrTimeout 等待超时. 0 s6 j0 k4 T& r$ j: o
wrAbandoned 在该事件的超时期限到达前,该事件对象已经被毁灭了。.
! n: h# |. f! o8 G! O* AwrError 在等待过程中有异常产生,要知道具体产生的错误要查看 TEvent的LastError
, u7 V8 r( i: u: o- P) J属性。
3 E5 x) {" B( F/ ]! G4 _2 ` 5 k0 `7 s. v: t6 a& @4 t7 J/ ]
3? TCriticalSection
" m/ g6 `- v# H8 }5 G9 c- j这个相当于WIN32编程中的临界区。 3 |6 c ~ U3 M$ e! I
在多线程编程中,多个线程需要访问同一个公用变量的时候。 8 ~ O5 U. p0 o0 u* M; ^1 d7 r
U9 V, A# X) b( S+ ?
来保证访问的正确性。对公用变量访问的代码写在Enter();和Leave()之间。
, E4 O) k- P3 x3 s" f比如有个公用变量 Count;
0 |1 H. r4 W: _7 D1 ]9 k以下代码 :
1 n4 V v1 ?. U) l0 `TCriticalSection * pSection=new TCriticalSection();
1 @1 @+ V: s2 r* L: h- l pSection->Enter();
4 ~) P0 q) q2 J7 e- F3 o( M8 K/ A* z Count++;
! b9 o6 v/ `+ R; q% P$ S8 p8 } pSection->Leave();
- y9 @9 T5 Q9 Z7 {. A$ Ydelete p;
9 M5 k0 `3 D2 n0 c1 I9 G ( p y% x3 d; { ~
Enter()方法进入临界区,对其中的公用变量加锁。 # n' c0 P9 U% m5 S0 r; Q
Leave()方法离开临界区,对其中的公用变量解锁。
" @7 ?9 b' n& P! o; Z4 \5 H8 K- q
" q% K; D) M/ _! K: K- L: o 4 p* K' {- r0 q4 K! M" c8 V8 X
4.TMultiReadExclusiveWriteSynchronizer 1 ~+ z+ [3 f3 ] `* e0 F2 \
用来处理类似于多个生产者和多个消费者的问题。这里的消费者是指
! Q6 n& Q h* P! }7 b' M对公用变量进行读操作的线程。
' j& J4 E6 C9 N ~生产者是对公用变量进行写操作的线程。</P>
" B3 o. u- P. F4 q# X) o< >四个方法。
G- O. ?% k% K% S BeginRead0 F, o5 X9 Z5 F' s( B2 g
EndRead$ m5 D& i- b& ?
这两个方法用于消费者。( P3 ?# V6 ]4 M0 \$ J% s
BeginWrite
2 `0 A& G# ]$ l- j+ d9 S! Z; d8 P EndWrite/ U4 m) A. H$ V+ a* S
这两个方法用于生产者。</P> j' F2 N! K: q% [
< >使用的时候就是要把这个TMutiReadExclusiveWriteSynchronizer 定义一个全局变量。( L0 K) o, q5 W& |
然后在其他线程中访问他。</P>
, k O; n) \" k. @3 v) ` W# a ~+ k1 ?6 C+ v' b# B5 ?( e+ {
< >
* J( y/ k" j, U/ F* w2 A </P> |
zan
|