在之前的文章“讓WCF客戶端的“調用”成為一種‘享受’”中,分別用Func與Action實現了WCF客戶端調用接口。
Func實現的調用方式:
WcfClient.UseService((IUserService userService) => (userService.GetUser(userId)));
缺點:
1. 需要用匿名方法傳遞參數,寫起來比較麻煩。
2. 不支持無返回值的ServiceContract。
Action實現的調用方式:
List<ZzkDocument> docs = null;
WcfClient.UseService<IZzkDocumentService>(
s =>
{
docs = s.GetZzkDocuments("0", 30).ToList();
});
缺點:
雖然支持了無返回值的ServiceContract,但需要在Lambda表達式中獲取返回值。這樣還造成了無法使用類型判斷,也就是不能使用var docs。
用Action“享受”了很多次的過程中,總是有那么一點點“不爽”縈繞在心頭,揮之不去。。。
2011即將過去,2012即將到來,在這辭舊換新之際,也把那一點點“不爽”辭去吧。
今天下午,借助.NET世界中一個強大的武器完成了這個小小的心願,它就是表達式樹 —— Expression<Func<TService, TReturn>> operation
先一睹調用改進版WCF Client的風采:
var client = new WcfClient<IZzkDocumentService>();
var docs = client.UseService(s => s.GetZzkDocuments("0", 10));
然后看看WcfClient的實現代碼:
public class WcfClient<TService> where TService : class
{
public TReturn UseService<TReturn>(Expression<Func<TService, TReturn>> operation)
{
var channelFactory = new ChannelFactory<TService>("*");
TService channel = channelFactory.CreateChannel();
var client = (IClientChannel)channel;
client.Open();
TReturn result = operation.Compile().Invoke(channel);
try
{
if (client.State != CommunicationState.Faulted)
{
client.Close();
}
}
catch
{
client.Abort();
}
return result;
}
}
對於Expression<Func<TService, TReturn>> operation,我的理解是:請你告訴我,如果給你一段代碼(TService),你如何給我一個改變了的世界(TReturn)。我不關心給你的是什么代碼,也不關心改變后的世界是什么樣子,我只關心你如何改變世界。
享受無止境,但真正讓人成長與快樂的不是得到那個享受,而是這個追求享受的過程。現實世界如此,代碼世界也是如此。
祝大家2012年在代碼世界享受更多快樂!
參考文章: