当年在ASP.NET MVC 1.0时代我提到,在开发时最好将视图的Model定制为强类型的,这样可以充分利用静态检查功能进行排错。不过有人指出,这么做虽然易于静态检查,但是定义强类型的Model类型实在是太麻烦了,因此也出现了基于SmartBag等折衷方案。强类型是一种极端方案,而在C# 4.0中我们又可以使用另一个极端,那就是让Model成为dynamic类型,这样在视图中便可以完全自由地获取数据了。不过,在使用匿名对象的情况下视图会抛出奇怪的“无法找到成员”异常,我们必须解决这个问题。

dynamic类型的视图模型

我们现在先来创建一个Model类型为dynamic的视图,例如Views\Home\Index.aspx,我们要在一开始的Page标记中设置它的基类类型:

<%@ Page Language="C#" Inherits="System.Web.Mvc.ViewPage<dynamic>" %>

我们将这里的范型参数设为dynamic。这一做法可能会让某些朋友感到新奇,不过这其实十分正常。事实上,dynamic关键字在C#的概念中(不论实现),其实就是和int、string或是Controller一样,都是一种“类型”,只不过对于这样一个类型会有些动态分发等特殊处理。这样定义视图类型之后,便可以在Controller中自由提供视图模型了,例如:

public ActionResult Index()
{
dynamic model = new ExpandoObject();
model.Hello = "World"; return View(model);
}

ExpandoObject是.NET 4.0在BCL中提供的类,它的作用便是利用.NET中定义的动态分发功能,定义了一个可以任意扩展的类型。例如在上面的代码中,我们可以直接赋予它Hello属性。然后在视图模板中读出:

<%= Model.Hello %>

这样看起来是不是很方便?这样虽然没有了任何的静态教研,倒也带来了十足的自由性——就像是Ruby on Rails等动态语言的框架一样。

奇怪的“无法找到成员”异常

不过,这种dynamic有时候会有非常奇怪的表现,例如我们把上面Index的代码改成使用匿名类型对象之后:

public ActionResult Index()
{
return View(new { Hello = "World" });
}

再次执行就会抛出异常:

简单说来,便是指Model对象没有找到合适的Hello成员,因此绑定失败。但是,我们明明有这个属性,不是吗?更奇怪的是,如果是在一个Console应用程序中这么写则是没有任何问题的:

dynamic model = new { Hello = "World" };
Console.WriteLine(model.Hello);

问题似乎就是出在ASP.NET的页面上。说到原因,我也不知道,一开始我以为是由于只读属性造成的,但是在尝试了显式定义类型之后发现也不是这个原因。我相信仔细分析之后一定可以得出结果,不过现在最重要的是想个办法解决它。毕竟这个情况很容易遇到,ExpandoObject只能解决极小部分的问题。因为在Controller中的常见需求之一,便是使用如LINQ表达式等方式生成一些供视图使用的匿名对象。

生成动态类型

这个问题以前也有人遇到过,他的做法是为视图模型定义一个封装类。可惜这么做其实并没有解决问题,事实上它只解决了Model本身使用匿名对象的问题,但是如果是Model的某个字段返回一个匿名对象呢?再访问这个匿名对象还是有相同问题出现。幸好我们知道一个显示定义的类型是可以正常使用的,于是一个比较容易想到的方法便是在运行时生成动态类型。例如:

public static class DynamicFactory
{
private static ConcurrentDictionary<Type, Type> s_dynamicTypes = new ConcurrentDictionary<Type, Type>(); private static Func<Type, Type> s_dynamicTypeCreator = new Func<Type, Type>(CreateDynamicType); public static object ToDynamic(this object entity)
{
var entityType = entity.GetType();
var dynamicType = s_dynamicTypes.GetOrAdd(entityType, s_dynamicTypeCreator); var dynamicObject = Activator.CreateInstance(dynamicType);
foreach (var entityProperty in entityType.GetProperties())
{
var value = entityProperty.GetValue(entity, null);
dynamicType.GetField(entityProperty.Name).SetValue(dynamicObject, value);
} return dynamicObject;
} private static Type CreateDynamicType(Type entityType)
{
var asmName = new AssemblyName("DynamicAssembly_" + Guid.NewGuid());
var asmBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(asmName, AssemblyBuilderAccess.Run);
var moduleBuilder = asmBuilder.DefineDynamicModule("DynamicModule_" + Guid.NewGuid()); var typeBuilder = moduleBuilder.DefineType(
entityType.GetType() + "$DynamicType",
TypeAttributes.Public); typeBuilder.DefineDefaultConstructor(MethodAttributes.Public); foreach (var entityProperty in entityType.GetProperties())
{
typeBuilder.DefineField(entityProperty.Name, entityProperty.PropertyType, FieldAttributes.Public);
} return typeBuilder.CreateType();
}
}

我在这里定义了一个ToDynamic扩展方法,用于从任意对象扩展出一个……动态类型的对象。这个动态类型会根据输入对象中的属性信息,生成对应的公有字段,然后使用反射进行赋值。有了这个方法以后,遇到匿名类型也就只需稍多一步就够了:

return View(new { Hello = "World" }.ToDynamic());

即便像我之前所说的那样,使用LINQ语句为视图准备一些可用的匿名对象,也可以这样:

var categories = ...; // get categories;

dynamic model = new ExpandoObject();
model.PrivateCategories =
from c in categories
where c.AccessLevel == AccessLevel.Private
orderby c.IsDefault descending, c.Name
select new
{
CategoryID = c.NoteCategoryID,
CategoryName = c.Name,
IsDefault = c.IsDefault,
Count = c.NoteCount
}.ToDynamic(); model.PublicCategories =
from c in categories
where c.AccessLevel == AccessLevel.Public
orderby c.IsDefault descending, c.Name
select new
{
CategoryID = c.NoteCategoryID,
CategoryName = c.Name,
IsDefault = c.IsDefault,
Count = c.NoteCount
}.ToDynamic(); return View(model);

问题就这样解决了。

改进

很显然,上面的实现只是个雏形,其中最大的问题应该就是“性能”了。现在的代码中反复使用了反射,对此我们可以使用如Fast Reflection Library那样的方式来改善反射读写字段的执行效率。当然,可能最好的办法是动态生成出这样的代码了:

public class DynamicType
{
public DynamicType(object entity)
{
var strongTyped = (EntityType)entity; this.Abc = strongTyped.Abc;
this.Ijk = strongTyped.Ijk;
this.Xyz = strongTyped.Xyz;
...
}
}

这其实也很简单,有兴趣的话您也可以试试看。一会儿我也会去实现一下类似的功能,顺便尝试一下.NET 4.0——据说.NET 4.0的类库对生成动态方法较之前的版本有了更好的支持。

当类型为dynamic的视图模型遭遇匿名对象的更多相关文章

  1. .NET:关于数据模型、领域模型和视图模型的一些思考

    背景 数据模型.领域模型和视图模型是“模型”的三种角色,一些架构用一种类型表示这三种角色,如:传统三层架构.也有一些架构用两种类型表示这三种角色,如:结合ORM的领域驱动架构.非常少见的场景是用三种类 ...

  2. [转]架构蓝图--软件架构 "4+1" 视图模型

    架构蓝图--软件架构 "4+1" 视图模型 本文基于多个并发视图的使用情况来说明描述软件密集型系统架构的模型.使用多重视图允许独立地处理各"风险承担人":最终用 ...

  3. DDD:谈谈数据模型、领域模型、视图模型和命令模型

    背景 一个类型可以充当多个角色,这个角色可以是显式的(实现了某个接口或基类),也可以是隐式的(承担的具体职责和上下文决定),本文就讨论四个角色:数据模型.领域模型.视图模型和命令模型. 四个角色 数据 ...

  4. ASP.NET MVC with Entity Framework and CSS一书翻译系列文章之第三章:搜索、高级过滤和视图模型

    在这一章中,我们首先添加一个搜索产品的模块以增强站点的功能,然后使用视图模型而不是ViewBag向视图传递复杂数据. 注意:如果你想按照本章的代码编写示例,你必须完成第二章或者直接从www.apres ...

  5. ThinkPHP第十六天(redirect、join、视图模型)

    1.redirect /** * Action跳转(URL重定向) 支持指定模块和延时跳转 * access protected * @param string $url 跳转的URL表达式 * @p ...

  6. Windows Phone 8初学者开发—第12部分:改进视图模型和示例数据

    原文 Windows Phone 8初学者开发—第12部分:改进视图模型和示例数据 第12部分:改进视图模型和示例数据 原文地址:http://channel9.msdn.com/Series/Win ...

  7. 视图模型-Lambda表达式

    EF中通过改变实体对象达到操作数据库表数据的目的,在对数据库实体操作时,肯定少不了和Linq.Lambda打交道,熟悉SQL的话,上手 Linq并不难,from in where select... ...

  8. 领域模型(DomainModel)与视图模型(ViewModel)

    Model-View-Controller(模型-视图-控制器,MVC)模式将你的软件组织并分解成三个截然不同的角色: Model 封装了你的应用数据.应用流程和业务逻辑. View 从 Model ...

  9. 架构蓝图--软件架构 "4+1" 视图模型

    引言 我们已经看到在许多文章和书籍中,作者欲使用单张视图来捕捉所有的系统架构要点.通过仔细地观察这 些图例中的方框和箭头,不难发现作者努力地在单一视图中表达超过其表达限度的蓝图.方框是代表运行的程序吗 ...

随机推荐

  1. React学习笔记。

    有段时间没写博客了,最近这段时间有点说不出的苦,虽然说年轻的时候该多出去经历些事,但每次找工作东跑西跑,坐公交坐地铁浪费了我太多时间,我感觉到这是一种浪费,对生命的浪费.所以很想尽快找到一份工作,去努 ...

  2. SpringMVC一路总结(一)

    SpringMVC听闻已久,早在去年就被学长问到关于SpringMVC的基础知识,当时也没在意.主要是工作中也没有用到关于SpringMVC的技术,因此免于没有时间和精力的借口就没有接触和学习Spri ...

  3. Windows Live Writer 在线安装失败的解决方法。

    这里提供一种解决方法:下载离线安装包,我这个版本是2011的.大家有兴趣的话可以下载一下: http://wl.dlservice.microsoft.com/download/8/3/D/83D75 ...

  4. jQuery2.x源码解析(设计篇)

    jQuery2.x源码解析(构建篇) jQuery2.x源码解析(设计篇) jQuery2.x源码解析(回调篇) jQuery2.x源码解析(缓存篇) 这一篇笔者主要以设计的角度探索jQuery的源代 ...

  5. 用python实现最长公共子序列算法(找到所有最长公共子串)

    软件安全的一个小实验,正好复习一下LCS的写法. 实现LCS的算法和算法导论上的方式基本一致,都是先建好两个表,一个存储在(i,j)处当前最长公共子序列长度,另一个存储在(i,j)处的回溯方向. 相对 ...

  6. 学习EF之贪懒加载和延迟加载(2)

    通过昨天对EF贪婪加载和延迟加载的学习,不难发现,延迟加载还是很好用的,但是问题也就来了,有的时候我们只需要加载一个实体,不需要和他相关的外部实体,这时候我们来看看EF延迟加载时怎么作用的吧 打开pr ...

  7. Java核心技术点之多线程

    学习Java的同学注意了!!! 学习过程中遇到什么问题或者想获取学习资源的话,欢迎加入Java学习交流群,群号码:279558494 我们一起学Java! 本文主要从整体上介绍Java中的多线程技术, ...

  8. CentOS 6.7 如何启用中文输入法

    安装CentOS系统后,如何启用中文输入法呢?这个问题看起来简单,但对于Linux初学者来说,也可能不是一件容易的事.本文笔者和大家分享一下"CentOS 6.6 如何启用中文输入法&quo ...

  9. PALIN - The Next Palindrome 对称的数

    A positive integer is called a palindrome if its representation in the decimal system is the same wh ...

  10. 炫酷的jQuery对话框插gDialog

    js有alert,prompt和confirm对话框,不过不是很美体验也不是很好,用jQuery也能实现, 体验效果:http://hovertree.com/texiao/jquery/34/ 代码 ...