WCF單例服務(wù)中可擴(kuò)展性認(rèn)識(shí)
WCF開(kāi)發(fā)工具的出現(xiàn),為開(kāi)發(fā)人員帶來(lái)了非常大的好處。我們今天將會(huì)通過(guò)這篇文章中介紹的內(nèi)容為大家充分講解一下關(guān)于WCF單例服務(wù)的基本內(nèi)容,從而使朋友們可以加深對(duì)WCF工具的應(yīng)用。#t#
對(duì)并發(fā)服務(wù)調(diào)用請(qǐng)求的處理是WCF最基本要求,為了提供服務(wù)的響應(yīng)能力,WCF會(huì)在不同的線程中處理并發(fā)請(qǐng)求。在單例模式下,服務(wù)實(shí)例是唯一的,也就是說(shuō)相同的服務(wù)實(shí)例會(huì)同時(shí)被多個(gè)線程并發(fā)地訪問(wèn)。在默認(rèn)的情況下,多個(gè)線程以同步的方式訪問(wèn)WCF單例服務(wù)對(duì)象,也就是說(shuō),在某個(gè)時(shí)刻,最多只會(huì)有一個(gè)線程在使用服務(wù)實(shí)例。如果一個(gè)服務(wù)操作需要1秒,那么在一分鐘內(nèi)最多只能處理60個(gè)服務(wù)調(diào)用請(qǐng)求。倘若客戶端采用默認(rèn)的超時(shí)時(shí)限(1分鐘),對(duì)于60個(gè)并發(fā)地服務(wù)調(diào)用請(qǐng)求,至少會(huì)有一個(gè)服務(wù)調(diào)用會(huì)失敗。這極大地降低了WCF服務(wù)的可擴(kuò)展性、響應(yīng)能力和可用性。
為了讓讀者對(duì)WCF單例服務(wù)的低可擴(kuò)展性有一個(gè)深刻的認(rèn)識(shí),我寫(xiě)了一個(gè)極端的案例。從這個(gè)案例演示中,讀者會(huì)清晰地認(rèn)識(shí)到提供一個(gè)相同的功能,采用單調(diào)模式和單例模式,對(duì)客戶端影響的差別有多大。本案例同樣沿用計(jì)算服務(wù)的例子,Add方法中通過(guò)使線程休眠5秒模擬一個(gè)耗時(shí)的服務(wù)操作,下面是服務(wù)的定義,采用單調(diào)實(shí)例上下文模式。
- [ServiceBehavior(InstanceContextModeInstanceContextMode =
InstanceContextMode.PerCall)]- public class CalculatorService : ICalculator
- {
- public double Add(double x, double y)
- {
- Thread.Sleep(5000);
- return x + y;
- }
- }
在客戶端,通過(guò)ThreadPool模擬5個(gè)并發(fā)的客戶端,在Add操作調(diào)用成功后輸出當(dāng)前的時(shí)間,從而檢驗(yàn)服務(wù)的響應(yīng)能力。
- for (int i = 0; i <
- i++)
- {
- ThreadPool.QueueUserWorkItem(delegate
- {
- using (ChannelFactory<ICalculator> channelFactory =
new ChannelFactory<ICalculator>("calculatorservice"))- {
- ICalculator calculator = channelFactory.CreateChannel();
- Console.WriteLine("{3}: x + y = {2} when x = {0} and y = {1}",
1, 2, calculator.Add(1, 2), DateTime.Now); }- });
- }
從客戶端輸出結(jié)果我們可以看出,對(duì)于5個(gè)并發(fā)的服務(wù)調(diào)用均得到了及時(shí)的相應(yīng),這是我們希望看到的結(jié)果。
- 3/8/2009 08:03:17 : x + y = 3 when x = 1 and y = 2
- 3/8/2009 08:03:17 : x + y = 3 when x = 1 and y = 2
- 3/8/2009 08:03:17 : x + y = 3 when x = 1 and y = 2
- 3/8/2009 08:03:18 : x + y = 3 when x = 1 and y = 2
- 3/8/2009 08:03:18 : x + y = 3 when x = 1 and y = 2
但是,如果將實(shí)例上下文模式換成是InstanceContextMode.Single,情況就完全不一樣了。從最終的輸出結(jié)果可以看出,客戶端得到執(zhí)行結(jié)果的間隔為5s,由此可知服務(wù)操作在服務(wù)端是以同步的方式執(zhí)行的。
- [ServiceBehavior(InstanceContextModeInstanceContextMode =
InstanceContextMode.Single)]- public class CalculatorService : ICalculator, IDisposable
- {
- //省略實(shí)現(xiàn)
- }
輸出結(jié)果:
- 3/8/2009 08:03:25 : x + y = 3 when x = 1 and y = 2
- 3/8/2009 08:03:30 : x + y = 3 when x = 1 and y = 2
- 3/8/2009 08:03:35 : x + y = 3 when x = 1 and y = 2
- 3/8/2009 08:03:40 : x + y = 3 when x = 1 and y = 2
- 3/8/2009 08:03:45 : x + y = 3 when x = 1 and y = 2
WCF通過(guò)并發(fā)模式(Concurrency Mode)表示多線程訪問(wèn)WCF單例服務(wù)對(duì)象的方式,而并發(fā)模式作為一種服務(wù)行為可以通過(guò)ServiceBehaviorAttribute特性進(jìn)行設(shè)定。WCF通過(guò)ConcurrencyMode枚舉來(lái)表示不同形式的并發(fā)模式,三個(gè)枚舉值Single、Reentrant和Multiple分別表示單線程、重入和多線程三種并發(fā)模式。關(guān)于并發(fā)和并發(fā)模式,將在本書(shū)的下一卷予以詳細(xì)講解,在這里就不再作重復(fù)介紹了。ConcurrencyMode在ServiceBehaviorAttribute的定義如下:
- [AttributeUsage(AttributeTargets.Class)]
- public sealed class ServiceBehaviorAttribute :
Attribute, IServiceBehavior- {
- //其他成員
- public ConcurrencyMode ConcurrencyMode { get; set; }
- }
- public enum ConcurrencyMode
- {
- Single,
- Reentrant,
- Multiple
- }
ConcurrencyMode.Single是默認(rèn)采用的并發(fā)模式,這正是上面的例子中服務(wù)操作同步執(zhí)行的根本原因。為了讓服務(wù)操作異步地執(zhí)行,從未提供服務(wù)的響應(yīng)能力,我們只須要通過(guò)ServiceBehaviorAttribute將并發(fā)模式設(shè)為ConcurrencyMode.Multiple就可以了。
- [ServiceBehavior(InstanceContextModeInstanceContextMode =
InstanceContextMode.Single, ConcurrencyModeConcurrencyMode =
ConcurrencyMode.Multiple)]- public class CalculatorService : ICalculator, IDisposable
- {
- //省略實(shí)現(xiàn)
- }
輸出結(jié)果:
- 3/8/2009 08:05:05 : x + y = 3 when x = 1 and y = 2
- 3/8/2009 08:05:05 : x + y = 3 when x = 1 and y = 2
- 3/8/2009 08:05:05 : x + y = 3 when x = 1 and y = 2
- 3/8/2009 08:05:05 : x + y = 3 when x = 1 and y = 2
- 3/8/2009 08:05:06 : x + y = 3 when x = 1 and y = 2
如果將并發(fā)模式設(shè)為ConcurrencyMode.Multiple,意味著同一個(gè)服務(wù)實(shí)例在多個(gè)線程中被并發(fā)執(zhí)行。當(dāng)我們操作一些數(shù)據(jù)的時(shí)候,須要根據(jù)具體的情況考慮是否要采用一些加鎖機(jī)制來(lái)確保狀態(tài)的同步性。
WCF單例服務(wù)的基本概念就為大家介紹到這里。