在通過Remoting調用重載的泛型方法時,遇到了一個奇怪的問題,現使用一個例子一步步來說明如下。
一.沒有重載的情況
假設Remoting的接口是IComputer:
public interface IComputer
{
int Add<TEntity>(TEntity c);
}
在Remoting Server上的實現非常簡單:
public class Computer : IComputer
{
public int Add<TEntity>(TEntity c)
{
return 0;
}
}
然後,通過Spring.net分別在配置Server和Client的Remoting通道,接著Client作如下調用:
IComputer remoteComupter = (IComputer)Program.SpringContext.GetObject("remotingComputer");
int res2 = remoteComupter.Add<GameRecordDetail>(new GameRecordDetail());
這個調用是成功的,沒有任何問題,返回值為0。
二.添加一個重載
我們為IComputer接口添加一個重載方法,如下:
public interface IComputer
{
int Add<TEntity>(TEntity c);
int Add<TEntity>(IList<TEntity> list);
}
服務端對重載方法的實現仍然是直接返回0。這時在Client再次調用前面的方法,會拋出System.NullReferenceException。異常的堆棧位置是:
Server stack trace:
在 System.Runtime.Remoting.Messaging.MethodCall.ResolveMethod(Boolean bThrowIfNotResolved)
在 System.Runtime.Remoting.Messaging.MethodCall..ctor(Object handlerObject, BinaryMethodCallMessage smuggledMsg)
在 System.Runtime.Serialization.Formatters.Binary.BinaryMethodCall.ReadArray(Object[] callA, Object handlerObject)
在 System.Runtime.Serialization.Formatters.Binary.ObjectReader.Deserialize(HeaderHandler handler, __BinaryParser serParser, Boolean fCheck, Boolean isCrossAppDomain, IMethodCallMessage methodCallMessage)
在 System.Runtime.Serialization.Formatters.Binary.BinaryFormatter.Deserialize(Stream serializationStream, HeaderHandler handler, Boolean fCheck, Boolean isCrossAppDomain, IMethodCallMessage methodCallMessage)
在 System.Runtime.Remoting.Channels.CoreChannel.DeserializeBinaryRequestMessage(String objectUri, Stream inputStream, Boolean bStrictBinding, TypeFilterLevel securityLevel)
在 System.Runtime.Remoting.Channels.BinaryServerFormatterSink.ProcessMessage(IServerChannelSinkStack sinkStack, IMessage requestMsg, ITransportHeaders requestHeaders, Stream requestStream, IMessage& responseMsg, ITransportHeaders& responseHeaders, Stream& responseStream)
Exception rethrown at [0]:
在 System.Runtime.Remoting.Proxies.RealProxy.HandleReturnMessage(IMessage reqMsg, IMessage retMsg)
在 System.Runtime.Remoting.Proxies.RealProxy.PrivateInvoke(MessageData& msgData, Int32 type)
在 RemotingLib.IComputer.Add[TEntity](TEntity c)
三.通過修改方法名消除重載
現將後面添加的方法重命名為Add2,Client再次調用前面的方法,一切又恢復正常。
四.仍然使用重載,但去掉泛型參數
現在我們恢復重載,但將泛型參數去掉,如下所示:
public interface IComputer
{
int Add(GameRecordDetail c);
int Add(IList<GameRecordDetail> list);
}
服務端對兩個方法的實現仍然是直接返回0。Client再次調用前面的方法,一切是正常的。
五.測試的結果
根據上述的試驗,我們可以總結出,出現上述奇怪現象的條件有以下幾點:
1.通過Remoting調用的目標方法必須是泛型的。
2.通過Remoting調用的目標方法必須有重載。
如果滿足這兩個條件,那麼調用目標方法時就會拋出上面的System.NullReferenceException。那麼,為什麼會這樣了?答案探索中......
針對這個問題,目前我暫用的解決方案是,通過重命名方法名以消除重載。