异常信息:

"指定的架构无效。错误:
CLR 类型到 EDM 类型的映射不明确,因为多个 CLR 类型与 EDM 类型“Person”匹配。
以前找到的是 CLR 类型“A.Person”,
新找到的则是 CLR 类型“B.Person”。

这类异常信息在代码里面出现过几次,每次的解决方案都让人匪夷所思。不知道为什么出现,也不知道为什么被解决了。

查阅了一些国外的资料,链接

Don't use classes with the same name - EF uses only class names to identify the type mapped in EDMX (namespaces are ignored) - it is a convention to allow mapping classes from different namespaces to single model. The solution for your problem is to name your classes in BLL differently.

但实际情况是,这类异常不总是出现,而是在一个偶然的情况下出现。所谓偶然的情况,却是一种很普通又简单的调用。

using (var con = new MyContainer())
{
string sql = "select top 1 name from cat";
var p = con.Database.SqlQuery<contract.Dog>(sql).FirstOrDefault();
Console.WriteLine(p.Name);
}

查询cat数据,将第一个的name数据查询并存储在contract.Dog对象中。这类调用非常普通。

整理一下思路:

1. 出现"CLR 类型到 EDM 类型的映射不明确"异常,肯定是存在和ef模型中数据结构一样的类。

2. 与ef数据结构同名的类一直存在,但并非一直报错。

做以下测试:

测试一

在ef数据模型所在项目中(以下用entity表示),建立模型Person

同时,在entity项目中另外新建一个类,也命名为Person(当然名称空间不一样),属性一样(类型和名称)。

调试时,会发现报错,报错内容同上。

测试二

在测试一的基础上,去掉entity中手动创建的Person类,在解决方案下新建另一个项目contract,在contract中新建类,命名Person,属性同上。

调用代码:

using (var con = new MyContainer())
{
string sql = "select top 1 * from person";
var p = con.Database.SqlQuery<contract.Person>(sql).FirstOrDefault();
Console.WriteLine(p.Name);
}

单独运行这段时,不会报错。加上下面这段:

using (var con = new MyContainer())
{
string sql = "select top 1 * from person";
var p = con.Database.SqlQuery<efentity.Person>(sql).FirstOrDefault();
Console.WriteLine(p.Name);
}

执行完上面代码后,紧接着执行下面的代码。出现异常,异常同上。

加断点,跟踪con的数据明细。

在执行完第一段查询以前,con.base._internalContext.ObjectContext.MetadataWorkSpace._itemOCSpace.Value(以下简称MetaOCSpace)为空;

执行完第一段查询后,MetaOCSpace的数量出现32条,具体如下:

出现了Person的类型映射数据,此时Person类型映射到了contract.Person。

省略其他测试过程,有如下结论:

1. 当entity中出现同数据模型的类时,同类名同字段,无论什么时候用ef操作数据,都会报错。

2. 当entity所在的assembly没有同名类,但其他assembly(例contract)有同名类时。先有查询结果放入entity的任意类对象,后有查询结果放入contract的任意类对象时,就会报错。操作的先后顺序调换,结果一样。

这是因为DbContext的MetadataWorkSpace一旦生成会缓存起来。也就是说,在同一个应用程序域里面,一旦用dbcontext操作过数据库,它会自动读取类所在assembly里面的所有类,并尝试匹配数据库模型,然后将匹配结果保存起来(保存到上面的MetaOCSpace中)。当下次操作数据库时,返回数据对应类类所在其它assembly里面的类与当前已匹配数据库模型发生冲突时,便会报错。

3. 当client引用entity + client引用contract时,有结论2的隐患。而当entity引用contract,然后client引用entity时,同样存在问题。

这种情况一般出现在ef的枚举类型定义为引用外部类型(contract中定义的类型),这时就会出现entity引用contract,然后client引用entity的场景。配合以下代码:

using (var con = new MyContainer())
{
var p = con.Person.Where(pp => pp.Status == contract.We.One).FirstOrDefault();
Console.WriteLine(p.Name);
}

这时,也会出现报错。

解决思路:

1. 不要与entity中的模型同名,同字段。或者换过来entity中的模型加特殊标记

2. ef操作数据库时,返回数据的数据类型必须用entity项目中定义的类型。

以上内容,部分细节未仔细推敲,如有其他想法请留言。

异常跟踪之CLR 类型到 EDM 类型的映射不明确的更多相关文章

  1. 指定的架构无效。错误: CLR 类型到 EDM 类型的映射不明确

    在使用WebService开发时,同时使用了EF和linq,查询数据时,使用linq(查询订单)可以正常拉出数据, 但是使用EF(查询用户)却会报以下错误: {"指定的架构无效.错误: \r ...

  2. [CLR via C#]4. 类型基础及类型、对象、栈和堆运行时的相互联系

    原文:[CLR via C#]4. 类型基础及类型.对象.栈和堆运行时的相互联系 CLR要求所有类型最终都要从System.Object派生.也就是所,下面的两个定义是完全相同的, //隐式派生自Sy ...

  3. CLR:基元类型、引用类型和值类型

    最新更新请访问: http://denghejun.github.io   前言 今天重新看了下关于CLR基元类型的东西,觉得还是有必要将其记录下来,毕竟这是理解CLR成功 之路上的重要一步,希望你也 ...

  4. [CLR via C#]6. 类型和成员基础

    原文:[CLR via C#]6. 类型和成员基础 6.1 类型的各种成员 在一个类型中,可以定义0个或多个以下种类的成员: 1)常量    常量就是指出数据值恒定不变的符号.这些符号通常用于使代码更 ...

  5. [Clr via C#读书笔记]Cp5基元类型引用类型值类型

    Cp5基元类型引用类型值类型 基元类型 编译器直接支持的类型,基元类型直接映射到FCL中存在的类型. 作者希望使用FCL类型名称而避免使用关键字.他的理由是为了更加的清晰的知道自己写的类型是哪种.但是 ...

  6. 重温CLR(三)类型基础

    所有类型都从System.Object派生 “运行时”要求每个类型最终都要从System.Object类型派生.也就是说,一下两个类型的定义完全一致. //隐式派生自Object class Empl ...

  7. 【CLR Via C#】15 枚举类型与位类型

    1.基础 枚举类型(enumerated types)定义了一组“符号名称/值”配对. 枚举类型是值类型,每个枚举类型都是从System.Enum派生的,而System.Enum又是从System.V ...

  8. C#中的基元类型、值类型和引用类型

    C# 中的基元类型.值类型和引用类型 1. 基元类型(Primitive Type) 编译器直接支持的类型称为基元类型.基元类型可以直接映射到 FCL 中存在的类型.例如,int a = 10 中的 ...

  9. CLR-2-2-引用类型和值类型

    引用类型和值类型,是一个老生常谈的问题了.装箱拆箱相信也是猿猿都知,但是还是跟着CLR via C#加深下印象,看有没有什么更加根本和以前被忽略的知识点. 引用类型: 引用类型有哪些这里不过多赘述,来 ...

随机推荐

  1. 分享一张SQLSERVER执行流程的图片

    分享一张SQLSERVER执行流程的图片 有天论坛里有人问,一时间并发连接很多,是不是可以在SSMS里配置连接池 连接池是属于客户端的,配置只能在连接字符串里配置,修改你的连接字符串,SSMS没有一个 ...

  2. 跟我学STL系列(1)——STL入门介绍

    一.引言 最近这段时间一直都在自学C++,所以这里总结下自己这段时间的学习过程,通过这种方式来巩固自己学到的内容和以备后面复习所用,另外,希望这系列文章可以帮助到其他自学C++的朋友们. 由于本人之前 ...

  3. codepage IMLangCodePages

      http://baike.baidu.com/link?url=78DSTGAri8dvHNLQ03rThSKieJqhFwFWL4sQMao6cfaRSOUWN88QVBwmSJPCZch0vf ...

  4. XML相关知识全接触(一)

    XML文件格式已经出来很久了.他的风头如今在JSON.YAML等新兴文件格式的冲击下已经显的不那么强劲.但是XML仍然是当今世界上使用最广泛的文件格式.围绕着它也有一大堆的概念和知识点.所以我们还是很 ...

  5. django上传文件

    template html(模板文件): <form enctype="multipart/form-data" method="POST" action ...

  6. servlet/jsp GET/POST

    GET请求方式 当需要向服务器请求指定的资源时使用的方法 它不应该用于一些会造成副作用的操作中(在网络应用中用它来提交请求是一种常见的错误用法) 什么情况浏览器发送get请求 -在地址栏输入一个地址 ...

  7. servlet tomcat servlet debug常见错误404,405,500

    404 web服务器根据请求地址找不到对应资源 1,地址错误 2,web.xml文件中的两个<servlet-name>不一致 3,工程没有部署 4,web应用程序部署结构没有遵守serv ...

  8. paip . 解决spring No unique bean of type [com.mijie.homi.search.service.index.MoodUserIndexService]

    paip . 解决spring No unique bean of type   [com.mijie.homi.search.service.index.MoodUserIndexService] ...

  9. 固定表头/锁定前几列的代码参考[JS篇]

    引语:做有难度的事情,才是成长最快的时候.前段时间,接了一个公司的稍微大点的项目,急着赶进度,本人又没有独立带过队,因此,把自己给搞懵逼了.总是没有多余的时间来做自己想做的事,而且,经常把工作带入生活 ...

  10. 看大众点评V9新版如何为O2O止血 带领行业下半场回归理性

    前不久,美团点评CEO王兴提出的“中国互联网进入下半场”观点一直在持续发酵,并引发了整个互联网圈对于进入下半场该如何改革,如何迎战的深刻反思.在互联网的上半场,大家依托的是人口红利,但是到了下半场,用 ...