14

我有一面旗帜,例如

[Flags]
public enum DaysOfTheWeek
{
    Monday = 1,
    Tuesday = 2,
    Wednesday = 4,
    Thursday = 8,
    Friday = 16,
    Saturday = 32,
    Sunday = 64
}

如果我想使用 Linq 根据包含特定标志的变量进行过滤,我可以尝试在 lambda 语句中使用Enum.HasFlag来过滤多个标志,例如

DaysOfWeek weekendFilter = DaysOfTheWeek.Saturday | DaysOfTheWeek.Sunday;
var weekends = allDays.Where(d => d.DayOfWeek.HasFlag(weekendFilter));

这目前给出:

LINQ to Entities 无法识别方法 'Boolean HasFlag(System.Enum)' 方法,并且此方法无法转换为存储表达式。

4

3 回答 3

25

枚举标志基于二进制操作。

HasFlag = (allDays & (DaysOfTheWeek.Saturday | DaysOfTheWeek.Sunday)) != 0;

如果您愿意,可以将其抽象为扩展方法。

具体回答你的问题:

DaysOfWeek weekendFilter = DaysOfTheWeek.Saturday | DaysOfTheWeek.Sunday;
var weekends = allDays.Where(d => (d.DayOfWeek & weekendFilter) != 0);

我知道您可以将此谓词与 LINQ to SQL 一起使用。它将被翻译成 SQL(SQL Server 支持二进制算术就好了)。不确定 EF 是否具有与其 LINQ 支持相同的能力,通常远不如 L2S。

于 2012-09-06T19:17:21.093 回答
5

看起来它已在 EF 6.1 中修复。

但是,关于 EF Core,请检查.

于 2017-05-08T19:32:23.423 回答
0

该解决方案结合了@usr 的答案,以及对我正在使用的EntityFramework 版本的理解。EF4 不支持保存枚举,所以实体保存一个 int? 并且对象上的枚举属性未映射到 EF。枚举的属性将映射到其 get/set 的成员。

例如

public int? DayOfWeekValue { get; set; }
[NotMapped]
public DaysOfWeek DayOfWeek
{
    get { return (DaysOfWeek)DayOfWeekValue; }
    set { DayOfWeekValue= (int)value; }
}

正如@usr 的回答,过滤器需要通过二进制操作来完成。这可以通过将标志设置为扩展字节来完成

[Flags]
public enum DaysOfTheWeek : byte
{
    Monday = 1,
    Tuesday = 2,
    Wednesday = 4,
    Thursday = 8,
    Friday = 16,
    Saturday = 32,
    Sunday = 64
}

然后将值和过滤器转换为字节以执行二进制操作:

var weekends = allDays.Where(d => (((byte)d.DayOfWeekValue.Value) & (byte)weekendFilter) != 0);
于 2012-09-07T12:05:44.070 回答