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

我的地盘我做主
该用户从未签到
 |
< >最近在写一个程序用到了多线程,所以对CB下的多线程有一定的学习。
& B6 |! C$ n) G现在把自己的一些心得讲一下。水平有限,写的很粗略,请大家见谅。</P>
2 E8 H! U( z3 B; b; k0 F< >CB相对于VC来说,在CB下写多线程程序是很简单的。不仅是VCL中有TThread这个类。封装了那些关于多线程的WINDOW API。我觉得更方便的是他提供了# ?) X0 {# z5 Q9 ?6 C3 N- E3 ]
直接访问主VCL线程中对象的能力。可以很容易的和主线程中的窗体,控件. y4 V9 Y1 O5 R- J5 A5 i5 C: b
打交道。和单线程的方式没有太多区别。只是在有多个线程都要访问主线程
6 ]/ c" a- O# u7 t$ Z中的对象(比如访问同一个窗体上的StringGrid).只要用Thread的Synchronize方法来调用那段访问主VCL线程的代码(具体请看帮助),我们就不用担心访问冲突的问题了。而且对于多线程的同步和互斥,CB也对WINDOW 编程中那些机制进行了封装。比如对临界区CriticalSection封装为TCriticalSection.事件Event封装为TEvent.这些类相当简单好用。5 ], K: y7 I9 W- h
下面就是我觉得比较重要的几点,供大家参考.</P>1 I8 S9 U: _; t6 j/ g3 X1 X
< >
% N4 T1 H: ]# Y# l" D1。TThread的WaitFor方法。是等待一个线程返回。其返回值在这个线程里可以任意设定。以便在该线程返回的时候让调用他的线程知道他的运行情况。 0 r# }0 b! Q' B( T, N$ t. P. w$ Q
/ z- c( {- s: j0 J% _! q在TThread的 OnTerminate事件中做线程的清除工作。他不是线程运行的一部分。
0 Z% u8 a' I/ \; t2 ^; W而是主VCL线程的一部分。所以在其中不能访问Thread的局部变量(如 int __thread i)
! L+ I) @7 i8 d+ i, }你可以把清楚代码写在这里,不用管现在在EXCUTE()方法执行到了哪个地方。
7 q+ j$ C P0 {. f, D, g& L% a这么看起来有点类似于C++里的 finally 块的作用。
/ f. z U) h% y H' \$ W, I2 D) M, x) d4 {! O
2。TEvent很重要。实现线程的同步。WaitFor(int Timeout)功能类似于
# c, a; q% S4 e {8 N& s% S+ a$ IWINDOW API WaitforSingleObject().返回值包括: 3 g7 w1 p& Z- k3 |& S3 |
其中参数Timeout可以设为INFINITE表示永久等待,但这样,程序很容易死在这里。
* s H* i3 t( a0 C( x" ?
{# u5 G" ?8 b; X3 IwrSignaled 该事件发生(成功返回).
y( V4 q1 {0 f0 d1 L. T- I) YwrTimeout 等待超时.
- [* M5 m7 A/ T! A3 M$ N5 kwrAbandoned 在该事件的超时期限到达前,该事件对象已经被毁灭了。. % l5 H" M% E) G6 H( P# E! f: V3 l
wrError 在等待过程中有异常产生,要知道具体产生的错误要查看 TEvent的LastError
) z8 b6 f% P# C, \# k# S属性。
8 u) `; R' R! N2 v8 D; y
/ k$ ^2 ~4 ~+ f8 ?3? TCriticalSection # H/ L9 I* B; g9 A/ I- J. |
这个相当于WIN32编程中的临界区。 ( a. V w* t! P- h
在多线程编程中,多个线程需要访问同一个公用变量的时候。 , z$ ~5 _- F* t1 W4 b
4 ~ C$ c' J+ ?) ~来保证访问的正确性。对公用变量访问的代码写在Enter();和Leave()之间。
, Z# N @+ x6 U+ R" h比如有个公用变量 Count;
9 ?+ l( s# Q2 T. A; X8 u( Z以下代码 :
8 {+ V: i* t7 a* E* v' zTCriticalSection * pSection=new TCriticalSection();
% N; t+ X- Y) j, A* |( j pSection->Enter(); ! M7 w) n' S3 W( z [, f) @1 }1 z' T
Count++;
! I2 W. k& c0 w$ l% V* ~/ S pSection->Leave();
4 b E# r& y9 b; u/ {) D# Mdelete p; / q; d; g: i9 c! n6 R
" h; x# [, {* S- JEnter()方法进入临界区,对其中的公用变量加锁。 5 | m% t3 d9 ^8 l
Leave()方法离开临界区,对其中的公用变量解锁。
: D" b6 H) I2 \0 |+ {* b( x 8 f& F1 J8 v; w" _9 N* U4 u( ^
; K7 z) `7 Z: c: A4 i
4.TMultiReadExclusiveWriteSynchronizer
6 R1 ]1 f# o) z用来处理类似于多个生产者和多个消费者的问题。这里的消费者是指
) m1 ]1 W" ] U5 | ~+ u对公用变量进行读操作的线程。: m. C+ _8 t$ U, U
生产者是对公用变量进行写操作的线程。</P>0 ^ \. x0 H# S0 u/ J8 X r# C
< >四个方法。
9 D7 b5 X5 B8 {$ H0 b* a) ~) K, T BeginRead" [) k* d7 Z4 ~
EndRead, S4 y+ Q# `, u' T' {: E- J
这两个方法用于消费者。
9 H& p0 s% ]* b5 v$ [ BeginWrite
9 T1 u3 k" y' u( W) i, L( ^! Q EndWrite
" L" b4 \5 ]$ H, j" S3 s" j这两个方法用于生产者。</P>7 h$ Z4 \4 {" t' k: O: _7 `- [
< >使用的时候就是要把这个TMutiReadExclusiveWriteSynchronizer 定义一个全局变量。
1 D7 p' I0 W( Z3 j6 `然后在其他线程中访问他。</P>
) o" w: G6 F3 I e: H+ A/ L
$ Q4 C+ g8 {& t< >, \9 C- Q9 b% w7 x/ X" [ \) W
</P> |
zan
|