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

我的地盘我做主
该用户从未签到
 |
< >最近在写一个程序用到了多线程,所以对CB下的多线程有一定的学习。
/ {1 m' I1 D/ H5 A现在把自己的一些心得讲一下。水平有限,写的很粗略,请大家见谅。</P>
7 C2 O3 n) n9 Z6 J0 e( M< >CB相对于VC来说,在CB下写多线程程序是很简单的。不仅是VCL中有TThread这个类。封装了那些关于多线程的WINDOW API。我觉得更方便的是他提供了+ J. g! i+ ]) O: j& }. h
直接访问主VCL线程中对象的能力。可以很容易的和主线程中的窗体,控件
: ~9 C$ d" U; a打交道。和单线程的方式没有太多区别。只是在有多个线程都要访问主线程
! i& A# P. X% L中的对象(比如访问同一个窗体上的StringGrid).只要用Thread的Synchronize方法来调用那段访问主VCL线程的代码(具体请看帮助),我们就不用担心访问冲突的问题了。而且对于多线程的同步和互斥,CB也对WINDOW 编程中那些机制进行了封装。比如对临界区CriticalSection封装为TCriticalSection.事件Event封装为TEvent.这些类相当简单好用。+ a- s% t% m* w# b& S: R3 U+ f
下面就是我觉得比较重要的几点,供大家参考.</P>
8 ^; e% Z8 C* f9 Z) c+ ?0 P< >
3 h ^" f1 r8 a; |5 }1。TThread的WaitFor方法。是等待一个线程返回。其返回值在这个线程里可以任意设定。以便在该线程返回的时候让调用他的线程知道他的运行情况。
1 C3 {1 q6 l" K" a
! k* P5 L/ V9 `, ?在TThread的 OnTerminate事件中做线程的清除工作。他不是线程运行的一部分。
$ N5 c6 @; ~: ?2 p, @& E9 B而是主VCL线程的一部分。所以在其中不能访问Thread的局部变量(如 int __thread i) * A. E9 W& ^& m7 M0 E
你可以把清楚代码写在这里,不用管现在在EXCUTE()方法执行到了哪个地方。
8 C; ~0 r0 Y, Y( _" I$ A这么看起来有点类似于C++里的 finally 块的作用。 & o8 O- S. D9 j) R
& \8 a" w9 s- Y0 _2 D' X2。TEvent很重要。实现线程的同步。WaitFor(int Timeout)功能类似于 : F6 s7 x" | ~' ~) M
WINDOW API WaitforSingleObject().返回值包括: 9 b1 x& d# e" q* e# K/ k
其中参数Timeout可以设为INFINITE表示永久等待,但这样,程序很容易死在这里。
- B' p0 q9 a" s1 f* l$ [
+ J0 M+ f$ A- QwrSignaled 该事件发生(成功返回).
! u+ J* S1 P# n6 {: AwrTimeout 等待超时. : C! J- [' P- e5 G+ ^/ ^' X/ p
wrAbandoned 在该事件的超时期限到达前,该事件对象已经被毁灭了。.
" n; G% v( n" `) S3 a7 I: W* }wrError 在等待过程中有异常产生,要知道具体产生的错误要查看 TEvent的LastError
, t9 `, o- x2 l" c3 p7 g属性。
5 l; {. I. s) [' P/ G4 W 3 F, K/ p3 [6 r" z3 y
3? TCriticalSection
2 k p, [: _* W' N; D这个相当于WIN32编程中的临界区。
/ g6 G* n9 M% P在多线程编程中,多个线程需要访问同一个公用变量的时候。 1 K1 |1 M+ x5 {- S& A
8 o) @- k7 }' Z- \4 X" F1 j
来保证访问的正确性。对公用变量访问的代码写在Enter();和Leave()之间。
( I7 U( N* V+ H2 o比如有个公用变量 Count;
+ x* K/ P% I {0 Z. ]& _以下代码 : ' v$ K v, b0 A2 l
TCriticalSection * pSection=new TCriticalSection(); ( {& X* G5 s& M; K/ [
pSection->Enter();
6 S1 I0 l5 ~; t& T Count++;
5 q8 s! v, l7 N! q pSection->Leave();
1 P1 C" u, A3 j! j' O# P" Xdelete p;
1 h0 e; O8 g; w# \' R2 ? * N5 @5 F. q$ K$ j3 a' X8 b$ [2 ]
Enter()方法进入临界区,对其中的公用变量加锁。 - K" g- Q( u4 f
Leave()方法离开临界区,对其中的公用变量解锁。
3 c, D7 k6 e) A( B% v5 M, R/ f* F * S5 K5 w/ w2 l3 i4 q
}1 M/ u4 B, h3 c) s4.TMultiReadExclusiveWriteSynchronizer }1 G T$ ~" y
用来处理类似于多个生产者和多个消费者的问题。这里的消费者是指
, p4 x% O8 B5 A9 q1 ?7 n对公用变量进行读操作的线程。! c0 f4 T, E: w% Q0 V+ B* @/ D
生产者是对公用变量进行写操作的线程。</P>
$ p5 c0 w8 J% a0 ~& a< >四个方法。
6 d3 q" U: d" Z+ \1 x/ a' ]1 t$ a BeginRead
3 L- g# p) ]9 T' l( v EndRead
* g2 ?, L0 u* q, C( u这两个方法用于消费者。
; G0 ^" V# e& P: v2 O BeginWrite; x! [$ Q* M7 s8 v, Q7 ]1 m
EndWrite! |' ]" R0 u1 l
这两个方法用于生产者。</P>0 G! E" l4 a7 v y9 i @# G3 q7 t
< >使用的时候就是要把这个TMutiReadExclusiveWriteSynchronizer 定义一个全局变量。7 z- Q& J. `! |, ]0 z
然后在其他线程中访问他。</P>' a- F. n9 V% Y% l" L
% j1 u% n' X0 I# a
< >
" t1 _0 t+ W) U p+ Q: J, z' w </P> |
zan
|