在调用 RealProxy 基构造函数时,您传递要代理的目标对象的类型。我想要做的是动态地向代理类型添加接口(interface),以便可以将生成的代理类型转换为其他接口(interface)。
例如:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Reflection.Emit;
using System.Runtime.Remoting.Messaging;
using System.Runtime.Remoting.Proxies;
namespace ConsoleApplication17
{
class Program
{
static void Main(string[] args)
{
MyProxy<IFoo> proxy = new MyProxy<IFoo>(new Foo());
IFoo proxiedFoo = (IFoo)proxy.GetTransparentProxy();
// make a proxied call...
proxiedFoo.DoSomething();
// cast proxiedFoo to IDisposable and dispose of it...
IDisposable disposableFoo = proxiedFoo as IDisposable;
// disposableFoo is null at this point.
disposableFoo.Dispose();
}
}
}
public interface IFoo
{
void DoSomething();
}
public class Foo : IFoo, IDisposable
{
#region IFoo Members
public void DoSomething()
{
//
}
#endregion
#region IDisposable Members
public void Dispose()
{
// dispose
}
#endregion
}
public class MyProxy<T> : RealProxy where T : class
{
private T _target;
public MyProxy(T target) :
base(CombineType(typeof(T), typeof(IDisposable)))
{
this._target = target;
}
private static Type CombineType(Type type1, Type type2)
{
// How to implement this method, Reflection.Emit????
throw new NotImplementedException();
}
public override System.Runtime.Remoting.Messaging.IMessage Invoke(System.Runtime.Remoting.Messaging.IMessage msg)
{
return InvokeRemoteCall((IMethodCallMessage)msg, this._target);
}
/// <summary>
/// Invokes the remote call.
/// </summary>
/// <param name="methodCall">The method call.</param>
/// <param name="target">The target.</param>
/// <returns>A <see cref="ReturnMessage"/></returns>
private static IMessage InvokeRemoteCall(IMethodCallMessage methodCall, object target)
{
MethodInfo method = methodCall.MethodBase as MethodInfo;
object callResult = (target != null) ? method.Invoke(target, methodCall.InArgs) : null;
LogicalCallContext context = methodCall.LogicalCallContext;
var query = method.GetParameters().Where(param => ((ParameterInfo)param).IsOut);
ParameterInfo[] outParameters = query.ToArray();
return new ReturnMessage(callResult, outParameters, outParameters.Count(), context, methodCall);
}
}
}
因此,为了能够将代理类型转换为
IDisposable
,除了 IDisposable
之外,我还需要能够将 IFoo
发送到 RealProxy
基构造函数调用。本质上,我如何实现这个方法来动态添加
IDisposable
到 IFoo
以进行代理。private static Type CombineType(Type type1, Type type2)
{
// How to implement this method, Reflection.Emit????
throw new NotImplementedException();
}
最佳答案
有一种非常简单的内置方法可以实现这一目标。然而,如果人们还不知道它,它实际上是无法发现的:-)
为了能够控制从某个 RealProxy
派生类关联的\获得的透明代理上哪些转换操作是有效的,RealProxy
需要实现一个额外的接口(interface),即 IRemotingTypeInfo
。IRemotingTypeInfo
接口(interface)定义的方法之一是 bool CanCastTo(Type type, object o)
。每次尝试将代理对象转换为其他类型时都会调用此方法;可以通过 type
参数访问转换操作的“目标”类型。
因此,要允许您的代理“实现”多个接口(interface),只需从您希望支持的类型的 true
方法中返回 CanCastTo()
。
请注意,在转换之后,透明代理上的方法调用仍然由同一个 RealProxy
实例接收。
要进行更深入的讨论,您可以阅读这篇 MSDN 文章:Create a Custom Marshaling Implementation Using .NET Remoting and COM Interop
这是一个完整的例子:
interface IFaceA
{
void MethodA();
}
interface IFaceB
{
void MethodB();
}
class MultiFaceProxy : RealProxy, IRemotingTypeInfo
{
public MultiFaceProxy()
:base(typeof(IFaceA)) {}
public bool CanCastTo(Type fromType, object o)
{
return fromType == typeof(IFaceA) || fromType == typeof(IFaceB);
}
public string TypeName
{
get { return GetProxiedType().FullName; }
set { throw new NotSupportedException(); }
}
public override IMessage Invoke(IMessage msg)
{
// invoke logic
return null;
}
}
class Program
{
static void Main(string[] args)
{
MultiFaceProxy proxy = new MultiFaceProxy();
IFaceA ifa = (IFaceA) proxy.GetTransparentProxy();
// The following now also works thanks to CanCastTo()
IFaceB ifb = (IFaceB)ifa;
}
}
关于c# - 如何动态组合两个接口(interface)传递给RealProxy,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/18338259/