热爱测试,主要研究性能测试和自动化测试方面的技术,希望与同样对测试有热情的你一同进步成长

c#多线程的学习(2)(转)

上一篇 / 下一篇  2007-07-27 16:09:55 / 个人分类:软件开发

 
假设这样一种情况,两个线程同时维护一个队列,如果一个线程对队列中添加元素,而另外一个线程从队列中取用元素,那么我们称添加元素的线程为生产者,称取用元素的线程为消费者。生产者与消费者问题看起来很简单,但是却是多线程应用中一个必须解决的问题,它涉及到线程之间的同步和通讯问题。
ca A2[;N6q W3G0
arOr7l0
前面说过,每个线程都有自己的资源,但是代码区是共享的,即每个线程都可以执行相同的函数。但是多线程环境下,可能带来的问题就是几个线程同时执行一个函数,导致数据的混乱,产生不可预料的结果,因此我们必须避免这种情况的发生。C#提供了一个关键字lock,它可以把一段代码定义为互斥段(critical section),互斥段在一个时刻内只允许一个线程进入执行,而其他线程必须等待。在C#中,关键字lock定义如下:
lock(expression) statement_block
 
expression代表你希望跟踪的对象,通常是对象引用。一般地,如果你想保护一个类的实例,你可以使用this;如果你希望保护一个静态变量(如互斥代码段在一个静态方法内部),一般使用类名就可以了。而statement_block就是互斥段的代码,这段代码在一个时刻内只可能被一个线程执行。
twIa1F U9`wp)w0
  下面是一个使用lock关键字的典型例子,我将在注释里向大家说明lock关键字的用法和用途:
//lock.cs51Testing软件测试网 z LI;KqJ-B+\w
using System;51Testing软件测试网yB%k4d{7?
using System.Threading;

$L m*Sp.s051Testing软件测试网8_r3BH3P9}
internal class Account
Pn(Mj:})u0{51Testing软件测试网tbwn2~ c#Lk6i!W3o
  int balance;51Testing软件测试网:[y }]#|,[(P
  Random r = new Random();
H+P7]5j9G0
  internal Account(int initial)51Testing软件测试网/B7nyQi
  {51Testing软件测试网%{1s5\7@o LC3G:l
  balance = initial;51Testing软件测试网2aIpL vtWl? S
  }
C^6ICxJ3K?051Testing软件测试网#~9Oq#c.Q9vye l
  internal int Withdraw(int amount)51Testing软件测试网E6_3} u)@-XG
  {
$j"{-j?O0
  if (balance < 0)51Testing软件测试网5L#I9c2M2R3s]9q
  {51Testing软件测试网[} I @1s8L8GbnK8o
    file://如果balance小于0则抛出异常
L1tu Ax%h e%A\6?0
    throw new Exception("Negative Balance");51Testing软件测试网4L8K!X%Q8fu
  }
6H'lC&K Tj.yq U0
  //下面的代码保证在当前线程修改balance的值完成之前51Testing软件测试网Q8ak)^9Ak*ZH
  //不会有其他线程也执行这段代码来修改balance的值51Testing软件测试网3@.K;q"g4P,H?b$J
  //因此,balance的值是不可能小于051Testing软件测试网;}hB&X:]$?_
  lock (this)
R:D9`;Y#|@0
  {51Testing软件测试网3}e0t"tE7V rmgk |
    Console.WriteLine("Current Thread:"+Thread.CurrentThread.Name);
,Xxh0]!E3Yp3l@T0
    file://如果没有lock关键字的保护,那么可能在执行完if的条件判断之后
8j'eNMx_$X#si_&|%Tq0
    file://另外一个线程却执行了balance=balance-amount修改了balance的值51Testing软件测试网wD Leg/Nm
    file://而这个修改对这个线程是不可见的,所以可能导致这时if的条件已经不成立了
f$V toao!z_{0
    file://但是,这个线程却继续执行balance=balance-amount,所以导致balance可能小于0
"{!N"v V eR ke0
    if (balance >= amount)
R;d@ `_(Y LL0
    {
:J%]0x*tPw0
    Thread.Sleep(5);51Testing软件测试网-l(RV%IiC Y^ c
    balance = balance - amount;51Testing软件测试网+sh2]2k%@*xG
    return amount;
H U{'P*GS#A7RO0
    }51Testing软件测试网bo+NAg Q,z
    else51Testing软件测试网U? V4k:Wq \
    {
mo4v&D T},Xf~0
    return 0; // transaction rejected
(l"Xtp5E~,OY;tz0
    }51Testing软件测试网6r\om,xc
  }51Testing软件测试网{$Bb mvau*^
  }51Testing软件测试网`-[3I-WQ-m
  internal void DoTransactions()51Testing软件测试网`r4aX4l4N!k
  {
_.]f*hW,qBF0
  for (int i = 0; i < 100; i++)51Testing软件测试网&Lq!Mw(bT/dZs~
    Withdraw(r.Next(-50, 100));51Testing软件测试网J4c%i,dM
  }51Testing软件测试网sqK}{`8LVp
}
c\m$w,UYXe p({0
Mj hOn`[0internal class Test51Testing软件测试网0Pi.T;sI&c a
{51Testing软件测试网Jo R-r5\S%T r
  static internal Thread[] threads = new Thread[10];51Testing软件测试网Kh'dW9]
  public static void Main()
_d-vU5EM+|n\0
  {
Vod*M)B m!z X*U0
  Account acc = new Account (0);51Testing软件测试网@.wS3M,jr!Bt$c])W
  for (int i = 0; i < 10; i++)51Testing软件测试网i'e2b7o&Pg;}5~
  {51Testing软件测试网;N,fLK1A Ne
    Thread t = new Thread(new ThreadStart(acc.DoTransactions));
^*D&P1Wj ka0
    threads[i] = t;
#s E"np IL?2l0
  }
o(qb Y|^0
  for (inti = 0; i < 10; i++)51Testing软件测试网Y*Ay w'A7V5V'ew_#c-]
    threads[i].Name=i.ToString();
W T*N_L%S:p b0
  for (int i = 0; i < 10; i++)51Testing软件测试网*f}H])Y5a$Q7b6O*npp
    threads[i].Start();51Testing软件测试网*\FK^vH,j
  Console.ReadLine();51Testing软件测试网_MF,v-m:? rs
  }
/|n3[ {%HD0
}

8j%W)uK2oS051Testing软件测试网'\8wTt!Sp I
而多线程公用一个对象时,也会出现和公用代码类似的问题,这种问题就不应该使用lock关键字了,这里需要用到System.Threading中的一个类Monitor,我们可以称之为监视器,Monitor提供了使线程共享资源的方案。
4b*?@ W&_ edpCt051Testing软件测试网 i M%T.hMt/C"N(k
  Monitor类可以锁定一个对象,一个线程只有得到这把锁才可以对该对象进行操作。对象锁机制保证了在可能引起混乱的情况下一个时刻只有一个线程可以访问这个对象。Monitor必须和一个具体的对象相关联,但是由于它是一个静态的类,所以不能使用它来定义对象,而且它的所有方法都是静态的,不能使用对象来引用。下面代码说明了使用Monitor锁定一个对象的情形:
......51Testing软件测试网$gahTF` C'Y
Queue ōQueue=new Queue();
Kz1f-c#U9vR,u Gt&j0......
T.s@F/P J0Monitor.Enter(oQueue);
l!Q,keU?z0......//
现在oQueue对象只能被当前线程操纵了
!F K-l f,M0
Monitor.Exit(oQueue);//释放锁
51Testing软件测试网$yJ0BOr-e0S
如上所示,当一个线程调用Monitor.Enter()方法锁定一个对象时,这个对象就归它所有了,其它线程想要访问这个对象,只有等待它使用Monitor.Exit()方法释放锁。为了保证线程最终都能释放锁,你可以把Monitor.Exit()方法写在try-catch-finally结构中的finally代码块里。对于任何一个被Monitor锁定的对象,内存中都保存着与它相关的一些信息,其一是现在持有锁的线程的引用,其二是一个预备队列,队列中保存了已经准备好获取锁的线程,其三是一个等待队列,队列中保存着当前正在等待这个对象状态改变的队列的引用。当拥有对象锁的线程准备释放锁时,它使用Monitor.Pulse()方法通知等待队列中的第一个线程,于是该线程被转移到预备队列中,当对象锁被释放时,在预备队列中的线程可以立即获得对象锁。
Lc;XZq/P!S`8v051Testing软件测试网A I*Q,U5_&Mry
下面是一个展示如何使用lock关键字和Monitor类来实现线程的同步和通讯的例子,也是一个典型的生产者与消费者问题。这个例程中,生产者线程和消费者线程是交替进行的,生产者写入一个数,消费者立即读取并且显示,我将在注释中介绍该程序的精要所在。用到的系统命名空间如下:
using System;
7{.x2ap9_ J:T:YZz/O)P0using System.Threading;
首先,我们定义一个被操作的对象的类Cell,在这个类里,有两个方法:ReadFromCell()WriteToCell。消费者线程将调用ReadFromCell()读取cellContents的内容并且显示出来,生产者进程将调用WriteToCell()方法向cellContents写入数据。
public class Cell
0f%AA8i^lV0{
A}0_zc.Q0Z#L0
  int cellContents; // Cell对象里边的内容51Testing软件测试网 ?4D2V;yn@
  bool readerFlag = false; //状态标志,为true时可以读取,为false则正在写入
S"eOQN)];u ol0
  public int ReadFromCell( )
C:S.Fh] kC0
  {
q6vM4r s9IJ0
  lock(this) // Lock关键字保证了什么,请大家看前面对lock的介绍
l`p fT#d4c7`0
  {51Testing软件测试网!D0I$S ao9C Ea;[
    if (!readerFlag)//如果现在不可读取
(f5Q%U&b8E g0
    {
Sm?`.S@${0
    try
Jk)UT#O/c ew4B0
    {51Testing软件测试网\ {2NF7N:q
      file://等待WriteToCell方法中调用Monitor.Pulse()方法51Testing软件测试网} C8QO1h et
      Monitor.Wait(this);
[r"icG/l)h]0
    }
hm8T q0?z6]0
    catch (SynchronizationLockException e)
_O~A'l D%o|9L0
    {51Testing软件测试网!roj.`-tP
      Console.WriteLine(e);51Testing软件测试网7I Bj?%m:a5~s(G*J
    }
7A-u-b u9^#t#J&?0
    catch (ThreadInterruptedException e)51Testing软件测试网SF G VcQ*L eO
    {51Testing软件测试网 pJA r"Y!Qq:tXZ
      Console.WriteLine(e);
:f5U#rP]3W5Ps0
    }
&I%lSIk:r'lDz0
    }51Testing软件测试网S1AX'jv"lR
    Console.WriteLine("Consume: {0}",cellContents);51Testing软件测试网.Z,nCt-Jf#z8g
    readerFlag = false; file://重置readerFlag标志,表示消费行为已经完成
)VQ#DK2H]Af'Z0
    Monitor.Pulse(this); file://通知WriteToCell()方法(该方法在另外一个线程中执行,等待中)
3`^1Pp!Zdf-\0
  }51Testing软件测试网 rYbF7~
  return cellContents;
JEJwxEy0
  }51Testing软件测试网)Sz&s9B Vm.x

&D3GM3F @ t0M!?!N0
  public void WriteToCell(int n)51Testing软件测试网6| G]M~Jl1G
  {
[2~2[-Raeam+m0
  lock(this)51Testing软件测试网}-[%RH)E;Bc:W
  {51Testing软件测试网+R_^sQi&C$v
    if (readerFlag)51Testing软件测试网rDM3xG:W6K3c(X-t @$H'T
    {51Testing软件测试网p6vZ/Z#t'i
    try51Testing软件测试网#] Xx#M:~*I&eB
    {
{-X pK)m*M(S0
      Monitor.Wait(this);51Testing软件测试网6ff D"O3`6}/eE
    }
eB~(h_S0
    catch (SynchronizationLockException e)
/w9]\R.M k3X0
    {51Testing软件测试网h3o;wiP Mr
      file://当同步方法(指Monitor类除Enter之外的方法)在非同步的代码区被调用51Testing软件测试网+q*d6ETMX:Up
      Console.WriteLine(e);
!P4Wh k1sgh%l0
    }51Testing软件测试网l*s9\@&qs,Td
    catch (ThreadInterruptedException e)51Testing软件测试网 c M;wTa e.O!R'}y y
    {
1ci)Z5UDE X`c1PJ0
      file://当线程在等待状态的时候中止
X vz}6l(x-v0
      Console.WriteLine(e);51Testing软件测试网4c*q3v S/UU
    }51Testing软件测试网j1rF`Z8u7m_'A
    }
$xgH0SUM.t4r0
    cellContents = n;51Testing软件测试网)l$WT us:KD8c
    Console.WriteLine("Produce: {0}",cellContents);
o#cw+qhOJs0
    readerFlag = true;51Testing软件测试网`kBX%RI-m1g
    Monitor.Pulse(this); file://通知另外一个线程中正在等待的ReadFromCell()方法51Testing软件测试网AX#[9A%Zc1?T2G[
  }
z+qq9XQ2Bz0Bk$Px0
  }51Testing软件测试网4sE@(yI@ |H
}

lg+k%cUN0
下面定义生产者CellProd和消费者类CellCons,它们都只有一个方法ThreadRun(),以便在Main()函数中提供给线程的ThreadStart代理对象,作为线程的入口。
然后在下面这个类MonitorSampleMain()函数中我们要做的就是创建两个线程分别作为生产者和消费者,使用CellProd.ThreadRun()方法和CellCons.ThreadRun()方法对同一个Cell对象进行操作。
public class MonitorSample51Testing软件测试网CS @C.W/Q3R
{
9a'x(Lp[yd"R!}0
  public static void Main(String[] args)51Testing软件测试网1@/D'BXti
  {51Testing软件测试网 PN5T0zm Ao y0L
  int result = 0; file://一个标志位,如果是0表示程序没有出错,如果是1表明有错误发生
Th"]^,pP/|*d Y5h0
  Cell cell = new Cell( );
8H(uNS~ [P6X0
Ul/pS*_8I0
  //下面使用cell初始化CellProdCellCons两个类,生产和消费次数均为2051Testing软件测试网|'_wo_ oZ2`L2Y
  CellProd prod = new CellProd(cell, 20);51Testing软件测试网r5UZw3|%L-mhi
  CellCons cons = new CellCons(cell, 20);51Testing软件测试网]Gby)?Z

P&_l$L lI0
  Thread producer = new Thread(new ThreadStart(prod.ThreadRun));
Ch:?7OJd h._(?d0
  Thread consumer = new Thread(new ThreadStart(cons.ThreadRun));
4nx8@JN)p0
  //生产者线程和消费者线程都已经被创建,但是没有开始执行
ryD t i@a ZUq!m0
R&C JZ~ R&{K0
  try
$j2o k{ [ L*_6j0
  {
"H9S%_g0E(~/Q'W @S0
    producer.Start( );
5I e`.xr*b!z0
    consumer.Start( );
+J4o[j iQa0
r]/Zb2N0
    producer.Join( );
/]5l0[s(s Q0
    consumer.Join( );51Testing软件测试网aP_'tf3n d
    Console.ReadLine();51Testing软件测试网O`KbM*D
  }51Testing软件测试网'kh)j&IG0e _
  catch (ThreadStateException e)
"|b t2J0_TC0
  {51Testing软件测试网3}{ [0CIx7gz%b
    file://当线程因为所处状态的原因而不能执行被请求的操作51Testing软件测试网,bru6B? {'xt%y[
    Console.WriteLine(e);51Testing软件测试网j]M9~/D)oi
    result = 1;
c%|-M6T.t'f0zU0
  }
#fg|w0b-{o0
  catch (ThreadInterruptedException e)51Testing软件测试网8ZQ Sh3H)W"_,o e
  {
d(~Z A IK0
    file://当线程在等待状态的时候中止51Testing软件测试网3w1J/]8S*Iv,Jk
    Console.WriteLine(e);
6L4v7@;h!K"kw0
    result = 1;
$i4^AfX(A?0
  }
!t]8LL1w]_v;~0
  //尽管Main()函数没有返回值,但下面这条语句可以向父进程返回执行结果
k2\!x/f` MtAJ[0
  Environment.ExitCode = result;
^ fW.s,{th6r0
  }
f)swy gNvG!y0O0
}
public class CellProd51Testing软件测试网%qBDI4UR{Z?
{
Y"b o4jw)IA0
  Cell cell; //被操作的Cell对象
`BZRWW^Q `'\0
  int quantity = 1; //生产者生产次数,初始化为1
Oj3ra ID9m051Testing软件测试网Bgq*N{
  public CellProd(Cell box, int request)
us![Hv nd'}d6}/v0
  {51Testing软件测试网5A;{j7{CDk4c ps U
  //构造函数51Testing软件测试网I,U5nU&x)Q)t
  cell = box;51Testing软件测试网qo#Lpm
  quantity = request;51Testing软件测试网R){w,O2I1J.xH.L1F
  }
]SxR/nr2z"?wB0
  public void ThreadRun( )
nX U9k2C0
  {51Testing软件测试网)wP'B"pOO
  for(int looper=1; looper<=quantity; looper++)
&XQ~/]2g{0cl+k0
    cell.WriteToCell(looper); file://生产者向操作对象写入信息
5B:B_#I.K6l5y0
  }
#x ])Qsc7uVx0
}
Q'[I&H} B:k051Testing软件测试网:}?`-m;z cz-]*~
public class CellCons51Testing软件测试网E Kz }s5aBK t3el
{
wu2Z)n+D L6E$Nc0
  Cell cell;51Testing软件测试网e:} `(E{-O!O0{H?~*r
  int quantity = 1;51Testing软件测试网 `Wm,?`4HW;j

'O)\J3AK0
  public CellCons(Cell box, int request)
|fF+n#G#vf0
  {
Y%]RpY8e T0
  cell = box;
9~&T"pI&k q0
  quantity = request;51Testing软件测试网s!@'m;Q/H9X
  }51Testing软件测试网7p%Z9?Y}4gGjC
  public void ThreadRun( )
1{Yx1]5c4L0
  {51Testing软件测试网FWx9omhN c
  int valReturned;51Testing软件测试网s6G8dFaVQ
  for(int looper=1; looper<=quantity; looper++)
$DW{lbp8Dh.u0
    valReturned=cell.ReadFromCell( );//消费者从操作对象中读取信息51Testing软件测试网riE9Bu
  }51Testing软件测试网#Q.F8Y9V[M0V+L#g
}
大家可以看到,在上面的例程中,同步是通过等待Monitor.Pulse()来完成的。首先生产者生产了一个值,而同一时刻消费者处于等待状态,直到收到生产者的脉冲(Pulse)”通知它生产已经完成,此后消费者进入消费状态,而生产者开始等待消费者完成操作后将调用Monitor.Pulese()发出的脉冲。它的执行结果很简单:
Produce: 151Testing软件测试网Nf5j(dO-d2raB'X
Consume: 151Testing软件测试网$G(S Z-M,S s5N
Produce: 2
A&VdI:?0Consume: 251Testing软件测试网!e_4k1j_@:ef5`!lk\
Produce: 3
/Q7^cdd;K0Consume: 3
F?&}h^:{8`"]$J0...51Testing软件测试网g2l&B~0NJ;_F/pP
...
k};j+IW!JM0Produce: 2051Testing软件测试网 S-T,y lu:s6x Y7H
Consume: 20
事实上,这个简单的例子已经帮助我们解决了多线程应用程序中可能出现的大问题,只要领悟了解决线程间冲突的基本方法,很容易把它应用到比较复杂的程序中去。 

TAG: 软件开发

 

评分:0

我来说两句

Open Toolbar