【发布时间】:2020-05-02 08:08:26
【问题描述】:
我正在尝试找出以下内容之间的区别:
someListOfEnums.Cast<int>()
和
someListOfEnums.Select(a => (int)a)?
我发现前者在 Entity Framework Core 3.1 的 Where 子句中使用时会导致异常,但后者不会。我原以为他们会采取类似的行动。
举个例子: 公共枚举水果 { 苹果, 香蕉, 橘子 }
public class FruitTable
{
public int Id { get; set; }
public Fruit Value { get; set; }
}
public class FruitContext : DbContext
{
public DbSet<FruitTable> Fruit { get; set; }
}
public void TestMethod(FruitContext context)
{
var list = new List<Fruit>{Fruit.Apple, Fruit.Orange};
var breaks = list.Cast<int>();
var works = list.Select(a => (int)a);
var fruits1 = context.Fruit.Where(a => works.Contains(a.Value)).ToList(); //This works
var fruits2 = context.Fruit.Where(a => breaks.Contains(a.Value)).ToList(); //This breaks
}
似乎使用 .Cast<int>() 会导致 where 子句包含枚举的名称(Apple、Orange 等),而使用 .Select(a => (int)a) 则不会。
更新
我已经意识到我上面的示例不会导致同样的问题(抱歉)。我已经完成并创建了一个肯定会重现该问题的程序。
使用以下数据库:
CREATE DATABASE Fruit
USE Fruit
CREATE TABLE Fruit
(
Id INT NOT NULL PRIMARY KEY,
Value INT NOT NULL,
)
INSERT INTO Fruit VALUES (1, 0)
INSERT INTO Fruit VALUES (3, 2)
以下程序:
using Microsoft.EntityFrameworkCore;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;
namespace ConsoleApp
{
public class Program
{
static void Main(string[] args)
{
FruitTable.TestMethod(new FruitContext());
}
public enum Fruit
{
Apple,
Banana,
Orange
}
public class FruitTable
{
public int Id { get; set; }
public int Value { get; set; }
public static void TestMethod(FruitContext context)
{
IEnumerable<Fruit> list = new Fruit[] {Fruit.Apple, Fruit.Orange};
var breaks = list.Cast<int>();
var works = list.Select(a => (int) a);
var fruits1 = context.Fruit.Where(a => works.Contains(a.Value)).ToList(); //This works
var fruits2 = context.Fruit.Where(a => breaks.Contains(a.Value)).ToList(); //This breaks
}
}
public class FruitContext : DbContext
{
protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
{
optionsBuilder.UseSqlServer("Server=.;Database=fruit;Trusted_Connection=True;ConnectRetryCount=0");
}
public DbSet<FruitTable> Fruit { get; set; }
}
}
}
导致以下错误:
'无效的列名'橙色'。列名“Apple”无效。
编辑
只是补充一点,.Net Core 2.2 中不存在该问题,当我们迁移到 3.1 时出现。想一想——可能是因为这个:https://docs.microsoft.com/en-us/ef/core/what-is-new/ef-core-3.0/breaking-changes#linq-queries-are-no-longer-evaluated-on-the-client
【问题讨论】:
-
你遇到了什么异常?
-
我在错误信息中加入了一个更好的例子。
标签: c# linq .net-core casting ef-core-3.1