c# - 跨 AppDomain 边界代理匿名对象

标签 c# proxy razor appdomain anonymous-types

我正在尝试为 RazorEngine 解决的最受欢迎的功能之一vNext 支持在单独的 AppDomain 中加载模板程序集这样我们就可以在需要的时候卸载程序集。很好的功能请求,但它具有对可以注入(inject)模板的可能模型类型引入约束的链式 react 。

当前 v2.1 版本的一个很好的特性是能够使用匿名类型作为模型。我们做了确定模板中的模型类型是匿名类型的工作,我们将基本模板设置为 TemplateBase<dynamic> .运行时绑定(bind)器为我们处理模型成员的后期绑定(bind)调用。一切都很好。

当我们在单独的 AppDomain 中引入对运行模板的支持时我们现在有一个约束,即模型只能是 [Serializable] 的类型(这是通过继承 MarshalByRefObject 暗示的)。匿名类型不可序列化,并且也是 private .

我的想法是以某种方式在模板库中创建一个代理模型(声明为 dynamic ),它将调用发送到模型(它将在调用域中,而不是模板运行的域中)。本质上:

模板:

<h1>@Model.Name</h1>

调用@Model.Name会做类似的事情:

Template.Model (ModelProxy) -> GetMember(Name) -> |BOUNDARY| -> Model.Name

有没有人知道或体验过在另一个 dynamic 中尝试代理对匿名(或 AppDomain 对象)的调用的最佳方法? ?

重要的是,我并不是要通过 AppDomain 推送匿名对象边界,那是做不到的。

最佳答案

好的。假设您了解反射和创建新的 AppDomain。我知道你知道该怎么做......:)

我创建了两个辅助类,它们允许您传递匿名对象。 ProxyAnonymousObjectProxyDynamicObject。您在第一个 AppDomain 中创建 ProxyAnonymousObject 并在另一个 AppDomain 中使用 ProxyDynamicObject。 (这两个对象都存在于主要的 AppDomain 库中)

[Serializable]
public class ProxyAnonymousObject : ISerializable {

    static Dictionary<string, Type> cached = new Dictionary<string, Type>();

    object model;

    public Dictionary<string, object> ModelProperties = new Dictionary<string, object>();

    public ProxyAnonymousObject(object model) { this.model = model; }
    public ProxyAnonymousObject(SerializationInfo info, StreamingContext ctx) {
        try {

            string fieldName = string.Empty;
            object fieldValue = null;

            foreach (var field in info) {
                fieldName = field.Name;
                fieldValue = field.Value;

                if (string.IsNullOrWhiteSpace(fieldName))
                    continue;

                if (fieldValue == null)
                    continue;

                ModelProperties.Add(fieldName, fieldValue);

            }

        } catch (Exception e) {
            var x = e;
        }
    }

    public void GetObjectData(SerializationInfo info, StreamingContext context) {

        foreach (var pi in model.GetType().GetProperties()) {
            info.AddValue(pi.Name, pi.GetValue(model, null), pi.PropertyType);
        }

    }
}

public class ProxyDynamicObject : DynamicObject{
    internal ProxyAnonymousObject Proxy { get; set; }

    public ProxyDynamicObject(ProxyAnonymousObject model) {
        this.Proxy = model;
    }

    public override bool TryGetMember(GetMemberBinder binder, out object result) {
        result = Proxy.ModelProperties[binder.Name];
        return true;
    }
}

要让它在您的 MarshalByRefObject 继承类中工作,您只需将目标 动态对象 设置为等于 new ProxyDynamicObject(model)。在我写的示例中,我会这样调用。

instance = Activator.CreateInstance(type);
var setModel = type.GetMethod("SetModel", BindingFlags.Public | BindingFlags.Instance);
var render = type.GetMethod("Render", BindingFlags.Public | BindingFlags.Instance);

setModel.Invoke(instance, new object[] { new ProxyDynamicObject(model) });
render.Invoke(instance, null);

我已经写了一篇关于它的博文 http://buildstarted.com/2011/06/28/getting-anonymous-types-to-cross-the-appdomain-boundary/更详细地解释一下。 (虽然这是我不太擅长的事情)

这个实现肯定有问题。它不支持嵌套的匿名类型,我很确定它通常会崩溃。但这绝对能让您走上正轨。

关于c# - 跨 AppDomain 边界代理匿名对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6452034/

相关文章:

javascript - 如何获取 Asp.Net MVC 3 当前项目名称?

c# - 如何使用关键字作为属性名称?

用于数据保存的 Node.js 压缩代理

c# - 使用 linq 更新对象的性能更好

ruby-on-rails - 链接 Webrick 代理

nginx - 如何使用 NGINX 作为任何请求位置的转发代理?

c# - ASP MVC 文件上传 HttpPostedFileBase 为空

javascript - 单击链接时更改变量

c# - 在解决方案资源管理器中创建子项

xx.xx.xx 格式的 C# Timer 计数器