18

我有一个方法可以从存储过程中返回最小的结果来填充选择菜单。当我想要最简单的结果时,我将 bool getMin = true 传递给存储过程,当我想要完整记录时,我传递 bool getMin = false。

这导致“数据读取器与指定的不兼容”的实体框架错误

错误中最相关的部分

{"Message":"发生错误。","ExceptionMessage":"数据读取器与指定的 'CatalogModel.proc_GetFramingSystems_Result' 不兼容。类型的成员 'FrameType' 在同名数据读取器。","ExceptionType":"System.Data.EntityCommandExecutionException",

显然,错误告诉我,当数据读取器尝试设置不在查询结果中的属性“FrameType”时。

现在我明白了这个错误,我想知道的是我是否要把这个 sql sproc 分成两个 sprocs,或者有没有办法解决这个问题?

我的功能如下

public static IEnumerable<IFramingSystem> GetFramingSystems(int brandID, string frameType, string glazeMethod, bool getMin)
{
    using (CatalogEntities db = new CatalogEntities())
    {
        return db.proc_GetFramingSystems(brandID, frameType, glazeMethod, getMin).ToList<IFramingSystem>();
    };
}

下面是我的 TSQL

ALTER proc [Catelog].[proc_GetFramingSystems]
@BrandID   INT,
@FrameType VARCHAR(26),
@GlazeMethod VARCHAR(7) ='Inside',
@getMin    BIT = 0
as
BEGIN
SET NOCOUNT ON;
IF @getMin =0
BEGIN
SELECT c.ID,c.Name,c.Descr,c.FrameType,c.isSubFrame,
       c.GlassThickness,c.GlassPosition,c.GlazingMethod,c.SillProfile
        from Catelog.Component c
WHERE c.MyType ='Frame' 
AND c.FrameType = @FrameType
AND c.GlazingMethod = @GlazeMethod
AND c.ID IN(
SELECT cp.ComponentID FROM Catelog.Part p JOIN
            Catelog.ComponentPart cp ON p.ID = cp.PartID
            WHERE p.BrandID = @BrandID
            )
            ORDER BY c.Name
END
ELSE
SELECT c.ID,c.Name,c.Descr
        from Catelog.Component c
WHERE c.MyType ='Frame' 
AND c.FrameType = @FrameType
AND c.GlazingMethod = @GlazeMethod
AND c.ID IN(
SELECT cp.ComponentID FROM Catelog.Part p JOIN
            Catelog.ComponentPart cp ON p.ID = cp.PartID
            WHERE p.BrandID = @BrandID
            )
            ORDER BY c.Name
SET NOCOUNT OFF;
END;
4

3 回答 3

16

对我来说,似乎两个分支都IF返回不同的数据,第一个分支返回 9 列,而第二个 - 只有三列。我相信 EF 不能反映IFramingSystem后者。具体来说,该列FrameType(和其他 5 个列)显然缺失:

 ...
 SELECT c.ID,c.Name,c.Descr    <- where are the remaining columns
    from Catelog.Component c
 ...
于 2013-09-13T19:09:08.833 回答
12

我知道这是一个旧帖子;但是,我想分享我今晚在这方面学到的东西。我发现“错误消息中最相关的部分正在说明”就是这个。

db.proc_GetFramingSystems(brandID, frameType, glazeMethod, getMin).ToList<IFramingSystem>();

期望从存储过程返回一个别名为“FrameType”的列。

当我创建一个具有更多程序员友好名称的表的 POCO(普通旧 clr 对象)类时,我遇到了这个问题。我想要的是“EmailAddy”等等,而不是一个强类型的名称“email_address”。我创建了一个映射类,在其他映射列中说明了这一点。

this.Property(t => t.EmailAddy).HasColumnName("email_address");

尽管这是 EF 的其他部分工作所必需的,但在执行 db.SqlQuery 时不会引用映射类。所以,当下面的代码执行时

var a = new SqlParameter("@fshipno", shipno);
return _context.db.SqlQuery<EmailList>("exec spGetEmailAddy @fshipno", a).ToList();

它产生了相同的错误,除了它提到了“EmailAddy”而不是“FrameType”。修复...我必须将存储过程中的“email_address”列别名为“EmailAddy”,以使其将返回的数据集映射到我的 POCO。

编辑:我发现这仅在您的方法返回 IEnumberable 时才有效

public IEnumberable<myPOCO> GetMyPoco()

如果您尝试返回单个 POCO 对象,您将收到相同的错误消息。

public myPOCO GetMyPoco()
于 2016-10-17T03:25:40.870 回答
2

如果您正在插入/删除/更新(这些被 EF 视为“非查询”),并且可以由我们的代码使用

MyDbContext.Database.ExecuteSqlCommand(insert into Table (Col1,Col2) values (1,2));

但是,如果正在对原始 SQL 语句进行选择查询,则使用

MyDbContext.DbSet<Table_name>.SqlQuery(select * from table_name).ToList();

或者

MyDbContext.Database.SqlQuery(select * from table_name).ToList();

()

SqlQuery()函数在 EF 中,出于奇怪的原因,抛出异常插入/删除/更新操作。(抛出的异常是“类型的成员,在数据读取器中没有对应的同名列。”)但是如果您打开Sql Management Studio并检查条目,它实际上已经执行了操作。

仅供参考http://www.entityframeworktutorial.net/EntityFramework4.3/raw-sql-query-in-entity-framework.aspx

于 2013-10-11T07:23:13.693 回答