奇妙的动态代理:EF中返回的对象为什么序列化失败
2016-01-05 23:18
225 查看
今天有如鹏的学生遇到一个问题:把一个对象保存到Session中(进程外Session)后,Web服务器重启,当从Session读取这个对象的时候报错,提示是一个“T_Users”后面跟着一大串数字的类型,不是“T_Users”类型。
凭着感觉,我问“这个对象是普通对象还是什么对象”,回复说“是Entity Framework返回的对象”,瞬间我知道了:是延迟加载造成的。
下面写个程序验证一下。
数据库里建立两张表:一张T_Persons表,一张T_Dogs表,T_Dogs表中有一个MasterId字段是指向T_Persons的外键,表示“狗的主人是谁”。
然后创建数据库的Entity。然后编写下面的代码:
第二行代码的意思很显然是:获得T_Dogs中第一个对象的主人对象,然后第三行获得的是对象的类型,我们惊奇的发现打印结果是:System.Data.Entity.DynamicProxies.T_Persons_A2C5F62323A22039FDCBEB40BD……
很奇怪,不应该是T_Persons类的对象吗,怎么是这么一个怪玩意。
这就涉及到“延迟加载(LazyLoading)”的原理了。对于需要延迟加载的对象,EF会动态生成一个实体类型的子类,这个子类是一个“动态代理(DynamicProxy)”类,这个类会进行延迟加载的处理。所以我们拿到的对象不直接是T_Persons类的对象,而是“动态代理(DynamicProxy)”类的子类,当然由于这个类是T_Persons的子类,所以“T_Persons p = e.T_Dogs.First().T_Persons”这样用父类的变量指向它也是可以的。多态经常偷偷帮我们做很多奇妙的事情。
由于这个对象是动态产生的类的对象,当程序重启之后,产生的“动态代理(DynamicProxy)”类和之前的“动态代理(DynamicProxy)”类就是两个类了。所以就会反序列化失败了。
很多ORM框架中延迟加载都是使用动态代理来实现的,比如Java中的Hibernate或者.Net的移植版NHibernate。
凭着感觉,我问“这个对象是普通对象还是什么对象”,回复说“是Entity Framework返回的对象”,瞬间我知道了:是延迟加载造成的。
下面写个程序验证一下。
数据库里建立两张表:一张T_Persons表,一张T_Dogs表,T_Dogs表中有一个MasterId字段是指向T_Persons的外键,表示“狗的主人是谁”。
然后创建数据库的Entity。然后编写下面的代码:
abcEntities e = new abcEntities(); T_Persons p = e.T_Dogs.First().T_Persons; Type t = p.GetType(); Console.WriteLine("对象类型:"+t);
第二行代码的意思很显然是:获得T_Dogs中第一个对象的主人对象,然后第三行获得的是对象的类型,我们惊奇的发现打印结果是:System.Data.Entity.DynamicProxies.T_Persons_A2C5F62323A22039FDCBEB40BD……
很奇怪,不应该是T_Persons类的对象吗,怎么是这么一个怪玩意。
这就涉及到“延迟加载(LazyLoading)”的原理了。对于需要延迟加载的对象,EF会动态生成一个实体类型的子类,这个子类是一个“动态代理(DynamicProxy)”类,这个类会进行延迟加载的处理。所以我们拿到的对象不直接是T_Persons类的对象,而是“动态代理(DynamicProxy)”类的子类,当然由于这个类是T_Persons的子类,所以“T_Persons p = e.T_Dogs.First().T_Persons”这样用父类的变量指向它也是可以的。多态经常偷偷帮我们做很多奇妙的事情。
由于这个对象是动态产生的类的对象,当程序重启之后,产生的“动态代理(DynamicProxy)”类和之前的“动态代理(DynamicProxy)”类就是两个类了。所以就会反序列化失败了。
很多ORM框架中延迟加载都是使用动态代理来实现的,比如Java中的Hibernate或者.Net的移植版NHibernate。
相关文章推荐
- 私有锁对象模式
- java中的GregorianCalendar类
- 如何安装SQL Server 2008数据库(带完整图解)
- MySQL 当记录不存在时insert,当记录存在时update(ON DUPLICATE KEY UPDATE, REPLACE)
- vs2010入门程序和出错问题解决方案
- jQuery UI API - 工具提示框部件(Tooltip Widget)
- 汇总遇到的一些小问题
- leetcode 198: House Robber
- CSDN博客支持数学公式了!
- android如何快速的浏览本地api。
- jQuery UI API - 工具提示框部件(Tooltip Widget)
- Effective Objective-C 2.0 第12条:理解消息转发机制
- 网络文章影响力挖掘
- Docker网络管理及容器跨主机通信(四)
- hdu 3939(勾股+容斥)
- hdu 3939(勾股+容斥)
- UVA10118(记忆化搜索 + 好题)
- 干货分享:详解线程的开始和创建
- 1030
- 2016-1-5-Thinking in Java 读书笔记(二) --一切都是对象