ÓÑÇéÌáʾ£ºÈç¹û±¾ÍøÒ³´ò¿ªÌ«Âý»òÏÔʾ²»ÍêÕû£¬Çë³¢ÊÔÊó±êÓÒ¼ü¡°Ë¢Ð¡±±¾ÍøÒ³£¡
µÚÈýµç×ÓÊé ·µ»Ø±¾ÊéĿ¼ ¼ÓÈëÊéÇ© ÎÒµÄÊé¼Ü ÎÒµÄÊéÇ© TXTÈ«±¾ÏÂÔØ ¡ºÊղص½ÎÒµÄä¯ÀÀÆ÷¡»

VB2008´ÓÈëÃŵ½¾«Í¨(PDF¸ñʽӢÎÄ°æ)-µÚ95²¿·Ö

¿ì½Ý²Ù×÷: °´¼üÅÌÉÏ·½Ïò¼ü ¡û »ò ¡ú ¿É¿ìËÙÉÏÏ·­Ò³ °´¼üÅÌÉ쵀 Enter ¼ü¿É»Øµ½±¾ÊéĿ¼ҳ °´¼üÅÌÉÏ·½Ïò¼ü ¡ü ¿É»Øµ½±¾Ò³¶¥²¿! Èç¹û±¾ÊéûÓÐÔĶÁÍ꣬ÏëÏ´μÌÐø½Ó×ÅÔĶÁ£¬¿ÉʹÓÃÉÏ·½ "Êղص½ÎÒµÄä¯ÀÀÆ÷" ¹¦ÄÜ ºÍ "¼ÓÈëÊéÇ©" ¹¦ÄÜ£¡


¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Monitor¡£Exit£¨_queue£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡ThreadPool¡£QueueUserWorkItem£¨AddressOf¡¡QueueProcessor£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡action£¨£©¡¡

¡¡¡¡¡¡¡¡End¡¡Sub¡¡



¡¡¡¡¡¡¡¡Public¡¡Sub¡¡New£¨£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡ThreadPool¡£QueueUserWorkItem£¨AddressOf¡¡QueueProcessor£©¡¡

¡¡¡¡¡¡¡¡End¡¡Sub¡¡



¡¡¡¡¡¡¡¡Public¡¡Sub¡¡Invoke£¨ByVal¡¡toExec¡¡As¡¡Action£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Monitor¡£Enter£¨_queue£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡_queue¡£Enqueue£¨toExec£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Monitor¡£Pulse£¨_queue£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Monitor¡£Exit£¨_queue£©¡¡

¡¡¡¡¡¡¡¡¡¡End¡¡Sub¡¡

End¡¡Class¡¡



¡¡¡¡¡¡¡¡¡¡ThreadPoolProducerConsumer¡¡has¡¡a¡¡single¡¡public¡¡method£»¡¡Invoke£¨£©£»¡¡which¡¡is¡¡used¡¡in¡¡the¡¡¡¡

same¡¡fashion¡¡as¡¡the¡¡Windows¡£Forms¡¡Invoke£¨£©¡¡method¡£¡¡What¡¡makes¡¡the¡¡generic¡¡producer/consumer¡¡¡¡

work¡¡is¡¡its¡¡use¡¡of¡¡the¡¡Monitor¡¡synchronization¡¡class¡£¡¡¡¡

¡¡¡¡¡¡¡¡¡¡To¡¡understand¡¡how¡¡Monitor¡¡works¡¡in¡¡the¡¡producer/consumer¡¡context£»¡¡consider¡¡the¡¡overall¡¡¡¡

producer/consumer¡¡implementation¡£¡¡The¡¡consumer¡¡thread¡¡£¨QueueProcessor£¨£©£©¡¡executes¡¡¡¡

constantly£»¡¡waiting¡¡for¡¡items¡¡in¡¡the¡¡queue¡¡£¨_queue£©¡£¡¡To¡¡check¡¡the¡¡queue£»¡¡the¡¡Monitor¡£Enter£¨£©¡¡¡¡

method¡¡is¡¡called£»¡¡which¡¡says£»¡¡¡°I¡¡want¡¡exclusive¡¡control¡¡for¡¡a¡¡code¡¡block¡¡that¡¡ends¡¡with¡¡the¡¡method¡¡¡¡

call¡¡Monitor¡£Exit£¨£©¡£¡±¡¡To¡¡check¡¡the¡¡queue£»¡¡a¡¡Do¡¡While¡¡loop¡¡is¡¡started¡£¡¡The¡¡loop¡¡waits¡¡until¡¡there¡¡¡¡

is¡¡something¡¡in¡¡the¡¡queue¡£¡¡The¡¡thread¡¡could¡¡execute¡¡constantly£»¡¡waiting¡¡for¡¡something¡¡to¡¡be¡¡¡¡

added£»¡¡but¡¡while¡¡the¡¡thread¡¡is¡¡looping£»¡¡it¡¡has¡¡control¡¡of¡¡the¡¡lock¡£¡¡This¡¡means¡¡a¡¡producer¡¡thread¡¡¡¡

cannot¡¡add¡¡anything¡¡to¡¡the¡¡queue¡£¡¡

¡¡¡¡¡¡¡¡¡¡The¡¡consumer¡¡needs¡¡to¡¡give¡¡up¡¡the¡¡lock£»¡¡but¡¡also¡¡needs¡¡to¡¡check¡¡if¡¡anything¡¡is¡¡available¡¡in¡¡¡¡

the¡¡queue¡£¡¡The¡¡solution¡¡is¡¡to¡¡call¡¡Monitor¡£Wait£¨£©£»¡¡which¡¡causes¡¡the¡¡consumer¡¡thread¡¡to¡¡release¡¡¡¡

the¡¡lock¡¡and¡¡say£»¡¡¡°Hey£»¡¡I¡¯m¡¡giving¡¡up¡¡the¡¡lock¡¡temporarily¡¡until¡¡somebody¡¡gives¡¡me¡¡a¡¡signal¡¡to¡¡¡¡

continue¡¡processing¡£¡±¡¡When¡¡the¡¡consumer¡¡thread¡¡releases¡¡its¡¡lock¡¡temporarily£»¡¡it¡¡goes¡¡to¡¡sleep¡¡¡¡

waiting¡¡for¡¡a¡¡pulse¡£¡¡


¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­Page¡¡388¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­

366¡¡¡¡¡¡¡¡¡¡¡¡¡¡CH¡¡AP¡¡T¡¡E¡¡R¡¡¡¡¡¡1¡¡3¡¡¡¡¡¡¡ö¡¡¡¡¡¡¡¡L¡¡E¡¡A¡¡R¡¡N¡¡I¡¡N¡¡G¡¡¡¡¡¡A¡¡B¡¡OU¡¡T¡¡¡¡¡¡M¡¡U¡¡L¡¡T¡¡I¡¡TH¡¡R¡¡E¡¡A¡¡DI¡¡N¡¡G¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡The¡¡producer¡¡thread¡¡£¨Invoke£¨£©£©¡¡also¡¡enters¡¡a¡¡protected¡¡block¡¡using¡¡the¡¡Monitor¡£Enter£¨£©¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡method¡£¡¡Within¡¡the¡¡protected¡¡block£»¡¡an¡¡item¡¡is¡¡added¡¡to¡¡the¡¡queue¡¡using¡¡the¡¡Enqueue£¨£©¡¡method¡£¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Because¡¡an¡¡item¡¡has¡¡been¡¡added¡¡to¡¡the¡¡queue£»¡¡the¡¡producer¡¡thread¡¡sends¡¡a¡¡signal¡¡using¡¡the¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Monitor¡£Pulse£¨£©¡¡method¡¡to¡¡indicate¡¡an¡¡item¡¡is¡¡available¡£¡¡This¡¡will¡¡cause¡¡the¡¡thread¡¡that¡¡gave¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡up¡¡the¡¡lock¡¡temporarily¡¡£¨the¡¡consumer¡¡thread£©¡¡to¡¡wake¡¡up¡£¡¡However£»¡¡the¡¡consumer¡¡thread¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡executes¡¡when¡¡the¡¡producer¡¡thread¡¡calls¡¡Monitor¡£Exit£¨£©¡£¡¡Until¡¡then£»¡¡the¡¡consumer¡¡thread¡¡is¡¡in¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡ready¡­to¡­execute¡¡mode¡£¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡In¡¡the¡¡simplest¡¡case¡¡of¡¡this¡¡implementation£»¡¡a¡¡single¡¡thread¡¡would¡¡constantly¡¡execute¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡QueueProcessor£¨£©¡£¡¡An¡¡optimization¡¡is¡¡to¡¡create¡¡and¡¡use¡¡a¡¡thread¡¡pool¡£¡¡A¡¡thread¡¡pool¡¡is¡¡a¡¡collec

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡tion¡¡of¡¡ready¡­to¡­execute¡¡threads¡£¡¡As¡¡tasks¡¡arrive£»¡¡threads¡¡are¡¡taken¡¡from¡¡the¡¡pool¡¡and¡¡used¡¡to¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡execute¡¡the¡¡tasks¡£¡¡Once¡¡the¡¡thread¡¡has¡¡pleted¡¡executing£»¡¡it¡¡is¡¡returned¡¡to¡¡the¡¡thread¡¡pool¡¡in¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡ready¡­to¡­execute¡¡mode¡£¡¡In¡¡the¡¡ThreadPoolProducerConsumer¡¡constructor£»¡¡the¡¡method¡¡ThreadPool¡£¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡QueueUserWorkItem£¨£©¡¡uses¡¡thread¡¡pooling¡¡to¡¡execute¡¡the¡¡method¡¡QueueProcessor£¨£©¡£¡¡In¡¡the¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡implementation¡¡of¡¡QueueProcessor£¨£©£»¡¡the¡¡method¡¡ThreadPool¡£QueueUserWorkItem£¨£©¡¡is¡¡called¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡again¡¡before¡¡calling¡¡the¡¡delegate¡£¡¡The¡¡result¡¡is¡¡that¡¡one¡¡thread¡¡is¡¡always¡¡waiting¡¡for¡¡an¡¡item¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡in¡¡the¡¡queue£»¡¡but¡¡there¡¡may¡¡be¡¡multiple¡¡threads¡¡executing¡¡concurrently£»¡¡processing¡¡items¡¡from¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡the¡¡queue¡£¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Using¡¡the¡¡generic¡¡producer/consumer¡¡is¡¡nearly¡¡identical¡¡to¡¡using¡¡the¡¡Windows¡£Forms¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Invoke£¨£©¡¡method¡£¡¡The¡¡following¡¡is¡¡a¡¡sample¡¡implementation¡£¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Imports¡¡System¡£Threading¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Public¡¡Class¡¡TestProducerConsumer¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Delegate¡¡Sub¡¡TestMethod£¨£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Sub¡¡Method£¨£©¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Console¡£WriteLine£¨¡¨Processed¡¡in¡¡thread¡¡id¡¡£¨¡¨¡¡&¡¡_¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Thread¡£CurrentThread¡£ManagedThreadId¡¡&¡¡¡¨£©¡¨£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡End¡¡Sub¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Public¡¡Sub¡¡TestSimple£¨£©¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Dim¡¡producer¡¡As¡¡ThreadPoolProducerConsumer¡¡=¡¡_¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡New¡¡ThreadPoolProducerConsumer£¨£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Console¡£WriteLine£¨¡¨Sent¡¡in¡¡thread¡¡id¡¡£¨¡¨¡¡&¡¡_¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Thread¡£CurrentThread¡£ManagedThreadId£¨£©&¡¡¡¨£©¡¨£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡producer¡£Invoke£¨AddressOf¡¡Method£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡End¡¡Sub¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡End¡¡Class¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡The¡¡TestSimple£¨£©¡¡method¡¡instantiates¡¡the¡¡ThreadPoolProducerConsumer¡¡type¡£¡¡Then¡¡the¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Invoke£¨£©¡¡method¡¡is¡¡called¡¡using¡¡the¡¡delegate¡¡TestMethod£»¡¡which¡¡executes¡¡the¡¡Method£¨£©¡¡method¡£¡¡¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡öNote¡¡¡¡The¡¡Thread¡£ManagedThreadId¡¡property¡¡uniquely¡¡identifies¡¡the¡¡managed¡¡thread¡¡to¡¡which¡¡it¡¡belongs¡£¡¡


¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­Page¡¡389¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡C¡¡HA¡¡P¡¡TE¡¡R¡¡¡¡¡¡1¡¡3¡¡¡¡¡¡¡ö¡¡¡¡¡¡¡¡L¡¡E¡¡AR¡¡N¡¡IN¡¡G¡¡¡¡¡¡AB¡¡O¡¡U¡¡T¡¡¡¡¡¡M¡¡U¡¡L¡¡T¡¡IT¡¡HR¡¡E¡¡AD¡¡IN¡¡G¡¡367¡¡



Using¡¡an¡¡Asynchronous¡¡Approach¡¡¡¡



Using¡¡asynchronous¡¡techniques¡¡means¡¡to¡¡perform¡¡a¡¡task£»¡¡such¡¡as¡¡read¡¡a¡¡file¡¡or¡¡database¡¡result£»¡¡¡¡

and¡¡then¡¡rather¡¡than¡¡wait¡¡for¡¡the¡¡results£»¡¡let¡¡some¡¡other¡¡code¡¡handle¡¡the¡¡results¡£¡¡The¡¡asynchro

nous¡¡interaction¡¡is¡¡an¡¡example¡¡of¡¡the¡¡producer/consumer¡¡architecture£»¡¡where¡¡the¡¡consumer¡¡¡¡

makes¡¡a¡¡request¡¡of¡¡the¡¡producer¡¡and¡¡carries¡¡on¡¡until¡¡it¡¡gets¡¡a¡¡response¡¡to¡¡that¡¡request¡£¡¡

¡¡¡¡¡¡¡¡¡¡The¡¡asynchronous¡¡technique¡¡used¡¡throughout¡¡the¡¡¡¡API¡¡is¡¡consistent¡¡and¡¡is¡¡easily¡¡¡¡

demonstrated¡¡by¡¡reading¡¡a¡¡file¡¡asynchronously¡£¡¡In¡¡Chapter¡¡10£»¡¡you¡¡learned¡¡how¡¡to¡¡read¡¡a¡¡file¡¡or¡¡¡¡

console¡¡stream¡¡using¡¡synchronous¡¡techniques¡£¡¡You¡¡could¡¡just¡¡as¡¡well¡¡have¡¡read¡¡that¡¡file¡¡or¡¡¡¡

stream¡¡asynchronously¡£¡¡

¡¡¡¡¡¡¡¡¡¡You¡¡would¡¡use¡¡asynchronous¡¡techniques¡¡when¡¡you¡¡don¡¯t¡¡want¡¡to¡¡wait¡¡around¡¡for¡¡the¡¡task¡¡¡¡

to¡¡plete¡£¡¡To¡¡read¡¡a¡¡file¡¡asynchronously£»¡¡the¡¡following¡¡source¡¡code¡¡is¡¡used¡£¡¡



Imports¡¡System¡£IO¡¡

Imports¡¡System¡£Text¡¡



Friend¡¡Module¡¡TestAsynchronousFile¡¡

¡¡¡¡¡¡¡¡Dim¡¡data¡¡As¡¡Byte£¨£©¡¡=¡¡New¡¡Byte£¨200000£©¡¡£û£ý¡¡

¡¡¡¡¡¡¡¡Private¡¡filename¡¡As¡¡String¡¡=¡¡¡¨threads¡£vb¡¨¡¡



¡¡¡¡¡¡¡¡Public¡¡Sub¡¡RunAll£¨£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡TestAsynchronousFile¡£LoadFileAsynchronously£¨£©¡¡

¡¡¡¡¡¡¡¡End¡¡Sub¡¡



¡¡¡¡¡¡¡¡Private¡¡Sub¡¡LoadFileAsynchronously£¨£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Dim¡¡fs¡¡As¡¡New¡¡FileStream£¨TestAsynchronousFile¡£filename£»¡¡FileMode¡£Open£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡fs¡£BeginRead£¨Data¡¡£»¡¡0£»¡¡Data¡¡¡£Length£»¡¡_¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡AddressOf¡¡DoAsyncRead£»¡¡fs£©¡£AsyncWaitHandle¡£WaitOne£¨£©¡¡

¡¡¡¡¡¡¡¡¡¡End¡¡Sub¡¡



¡¡¡¡¡¡¡¡Sub¡¡DoAsyncRead£¨ByVal¡¡result¡¡As¡¡IAsyncResult£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Dim¡¡localFS¡¡As¡¡FileStream¡¡=¡¡DirectCast£¨result¡£AsyncState£»¡¡FileStream£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Dim¡¡bytesRead¡¡As¡¡Integer¡¡=¡¡localFS¡£EndRead£¨result£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Dim¡¡buffer¡¡As¡¡String¡¡=¡¡Encoding¡£ASCII¡£GetString£¨data£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡Console¡£WriteLine£¨¡¨Buffer¡¡bytes¡¡read¡¡£¨¡¨¡¡&¡¡bytesRead¡¡&¡¡¡¨£©¡¨£©¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡localFS¡£Close£¨£©¡¡

¡¡¡¡¡¡¡¡End¡¡Sub¡¡

End¡¡Module¡¡



¡¡¡¡¡¡¡¡¡¡To¡¡read¡¡a¡¡file£»¡¡you¡¡need¡¡to¡¡open¡¡a¡¡file¡¡stream£»¡¡just¡¡as¡¡in¡¡Chapter¡¡10¡¯s¡¡examples¡£¡¡However£»¡¡¡¡

instead¡¡of¡¡reading¡¡the¡¡data¡¡directly£»¡¡the¡¡BeginRead£¨£©¡¡method¡¡is¡¡called£»¡¡and¡¡it¡¡starts¡¡a¡¡read¡¡operation¡£¡¡¡¡

What¡¡distinguishes¡¡the¡¡asynchronous¡¡operation¡¡is¡¡that¡¡BeginRead£¨£©¡¡returns¡¡immediately¡£¡¡Think¡¡of¡¡¡¡

it¡¡as¡¡starting¡¡the¡¡producer¡£¡¡

¡¡¡¡¡¡¡¡¡¡When¡¡you¡¡call¡¡BeginRead£¨£©£»¡¡the¡¡first¡¡three¡¡parameters¡¡represent¡¡the¡¡variable¡¡that¡¡contains¡¡¡¡

the¡¡read¡¡bytes¡£¡¡The¡¡first¡¡parameter¡¡is¡¡the¡¡byte¡¡array¡¡where¡¡the¡¡data¡¡should¡¡be¡¡written¡£¡¡The¡¡second¡¡¡¡

and¡¡third¡¡parameters¡¡are¡¡the¡¡starting¡¡and¡¡ending¡¡locations¡¡of¡¡the¡¡write¡¡operation¡¡in¡¡the¡¡byte¡¡¡¡

array¡£¡¡The¡¡fourth¡¡parameter¡¡to¡¡BeginRead£¨£©¡¡is¡¡the¡¡delegate¡¡that¡¡will¡¡be¡¡called¡¡when¡¡the¡¡data¡¡¡¡

is¡¡ready¡¡to¡¡be¡¡processed¡£¡¡The¡¡last¡¡parameter¡¡is¡¡like¡¡a¡¡thread¡¡parameter¡¡and¡¡is¡¡assigned¡¡to¡¡the¡¡¡¡


¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­Page¡¡390¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­¡­

368¡¡¡¡¡¡¡¡¡¡¡¡¡¡CH¡¡AP¡¡T¡¡E¡¡R¡¡¡¡¡¡1¡¡3¡¡¡¡¡¡¡ö¡¡¡¡¡¡¡¡L¡¡E¡¡A¡¡R¡¡N¡¡I¡¡N¡¡G¡¡¡¡¡¡A¡¡B¡¡OU¡¡T¡¡¡¡¡¡M¡¡U¡¡L¡¡T¡¡I¡¡TH¡¡R¡¡E¡¡A¡¡DI¡¡N¡¡G¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡IAsyncResult¡£AsyncState¡¡data¡¡member¡£¡¡IAsyncResult¡¡is¡¡in¡¡the¡¡System¡¡namespace¡¡and¡¡is¡¡there

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡fore¡¡part¡¡of¡¡the¡¡¡¡API¡£¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡When¡¡you¡¡call¡¡¡¡BeginRead£¨£©£»¡¡you¡¡are¡¡saying£»¡¡¡°Please¡¡fill¡¡up¡¡as¡¡many¡¡bytes¡¡as¡¡possible¡¡in¡¡the¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡byte¡¡array¡£¡¡When¡¡you¡¡have¡¡read¡¡the¡¡bytes£»¡¡call¡¡my¡¡consumer¡¡code¡£¡±¡¡The¡¡filling¡¡of¡¡the¡¡array¡¡and¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡calling¡¡of¡¡the¡¡consumer¡¡code¡¡occur¡¡in¡¡a¡¡separate¡¡thread¡£¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡For¡¡illustration¡¡purposes£»¡¡the¡¡AsyncWaitHandle¡£WaitOne£¨£©¡¡method¡¡is¡¡called¡¡so¡¡that¡¡the¡¡main¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡calling¡¡thread¡¡continues¡¡only¡¡after¡¡the¡¡data¡¡has¡¡been¡¡read¡¡and¡¡processed¡£¡¡It¡¡is¡¡a¡¡superfluous¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡method¡¡call£»¡¡since¡¡doing¡¡this¡¡would¡¡make¡¡the¡¡file¡­reading¡¡behavior¡¡resemble¡¡a¡¡synchronous¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡operation¡£¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡The¡¡consumer¡¡code¡¡£¨that¡¡is£»¡¡the¡¡DoAsyncRead£¨£©¡¡method£©is¡¡executed¡¡on¡¡another¡¡thread¡¡and¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡is¡¡responsible¡¡for¡¡reading¡¡all¡¡the¡¡bytes¡£¡¡The¡¡method¡¡receives¡¡an¡¡IAsyncResult¡¡parameter£»¡¡which¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡has¡¡an¡¡AsyncState¡¡property¡¡and¡¡is¡¡the¡¡value¡¡that¡¡we¡¡passed¡¡in¡¡as¡¡the¡¡fifth¡¡parameter¡¡to¡¡BeginRead£¨£©¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡£¨the¡¡file¡¡stream£©¡£¡¡DoAsyncRead£¨£©¡¡then¡¡converts¡¡this¡¡parameter¡¡into¡¡a¡¡¡¡FileStream¡¡and¡¡reads¡¡the¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡remaining¡¡bytes¡¡from¡¡the¡¡stream¡£¡¡The¡¡byte¡¡stream¡¡is¡¡then¡¡converted¡¡into¡¡a¡¡string¡¡buffer¡£¡¡When¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡all¡¡the¡¡data¡¡has¡¡been¡¡read£»¡¡the¡¡file¡¡stream¡¡is¡¡closed¡£¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡When¡¡using¡¡this¡¡asynchronous¡¡approach£»¡¡you¡¡are¡¡really¡¡writing¡¡producer/consumer¡¡code¡£¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡The¡¡use¡¡of¡¡¡¡IAsyncResult£»¡¡BeginRead£¨£©£»¡¡and¡¡EndRead£¨£©¡¡is¡¡quite¡¡mon¡£¡¡The¡¡purpose¡¡of¡¡the¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡asynchronous¡¡interface¡¡is¡¡to¡¡convert¡¡a¡¡synchronous¡¡operation¡¡into¡¡an¡¡asynchronous¡¡operation¡£¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡The¡¡Important¡¡Stuff¡¡to¡¡Remember¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡In¡¡this¡¡chapter£»¡¡you¡¡learned¡¡the¡¡essentials¡¡of¡¡using¡¡threads¡¡and¡¡how¡¡to¡¡write¡¡multithreaded¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡code¡£¡¡The¡¡main¡¡items¡¡to¡¡remember¡¡are¡¡as¡¡follows£º¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡o¡¡¡¡The¡¡operating¡¡system¡¡assigns¡¡time¡¡slices¡¡to¡¡your¡¡application¡£¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡o¡¡¡¡A¡¡time¡¡slice¡¡is¡¡a¡¡predefined¡¡amount¡¡of¡¡time¡¡in¡¡which¡¡your¡¡program¡¡can¡¡execute¡¡and¡¡has¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡full¡¡control¡¡of¡¡the¡¡microprocessor¡£¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡o¡¡¡¡To¡¡implement¡¡tasks¡¡that¡¡you¡¡want¡¡to¡¡run¡¡separately¡¡from¡¡the¡¡main¡¡program£»¡¡you¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡use¡¡threads¡£¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡o¡¡¡¡Each¡¡program¡¡that¡¡is¡¡started¡¡is¡¡a¡¡task¡¡and¡¡has¡¡a¡¡main¡¡thread¡¡from¡¡which¡¡you¡¡can¡¡start¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡other¡¡threads¡£¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡o¡¡¡¡Threading¡¡is¡¡not¡¡difficult¡¡and¡¡easily¡¡implemented¡£¡¡What¡¡is¡¡more¡¡difficult¡¡is¡¡synchroniza

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡tion¡¡between¡¡the¡¡threads¡£¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡o¡¡¡¡Synchronization¡¡is¡¡not¡¡about¡¡the¡¡data£»¡¡but¡¡about¡¡synchronizing¡¡access¡¡to¡¡code¡¡that¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡modifies¡¡data¡¡that¡¡is¡¡shared¡£¡¡If¡¡your¡¡data¡¡is¡¡not¡¡shared£»¡¡you¡¡don¡¯t¡¡need¡¡synchronization¡£¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡o¡¡¡¡In¡¡the¡¡context¡¡of¡¡a¡¡single¡¡application£»¡¡you¡¡will¡¡use¡¡either¡¡the¡¡exclusive¡¡lock¡¡or¡¡the¡¡Monitor¡¡¡¡

¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡for¡¡synchronization¡£¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡o¡¡¡¡Locking¡¡code¡¡slows¡¡down¡¡the¡¡code¡£¡¡You¡¡should¡¡keep¡¡the¡¡locks¡¡for¡¡the¡¡shortest¡¡time¡¡possible¡£¡¡



¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡¡o¡¡¡¡To¡¡improve¡¡throughput£»¡¡you¡¡can¡¡take¡¡a¡¡snapshot¡¡of¡¡the¡¡data¡£¡¡

·µ»ØĿ¼ ÉÏÒ»Ò³ ÏÂÒ»Ò³ »Øµ½¶¥²¿ ÔÞ£¨0£© ²È£¨0£©
¿ì½Ý²Ù×÷: °´¼üÅÌÉÏ·½Ïò¼ü ¡û »ò ¡ú ¿É¿ìËÙÉÏÏ·­Ò³ °´¼üÅÌÉ쵀 Enter ¼ü¿É»Øµ½±¾ÊéĿ¼ҳ °´¼üÅÌÉÏ·½Ïò¼ü ¡ü ¿É»Øµ½±¾Ò³¶¥²¿!
ÎÂÜ°Ìáʾ£º ο´Ð¡ËµµÄͬʱ·¢±íÆÀÂÛ£¬Ëµ³ö×Ô¼ºµÄ¿´·¨ºÍÆäËüС»ï°éÃÇ·ÖÏíÒ²²»´íŶ£¡·¢±íÊéÆÀ»¹¿ÉÒÔ»ñµÃ»ý·ÖºÍ¾­Ñé½±Àø£¬ÈÏÕæдԭ´´ÊéÆÀ ±»²ÉÄÉΪ¾«ÆÀ¿ÉÒÔ»ñµÃ´óÁ¿½ð±Ò¡¢»ý·ÖºÍ¾­Ñé½±ÀøŶ£¡