LINQ查询中奇怪的扩展方法行为

Moh*_*deh 1 c# linq extension-methods enums

我有一个自定义属性存储枚举项的显示名称.我写了一个返回Enum自定义属性显示名称的Extension方法.

我在LINQ查询中使用我的Extension方法.但输出很奇怪.当我在一个尚未运行的查询中调用我的扩展方法时,它返回Enum is null但是当我在.ToList()调用的查询中运行它时,它工作正常(请参阅输出)...它们之间有什么不同?当我的查询没有运行时,为什么enum为null

注意:我是LINQPAD.

void Main()
{
    Console.WriteLine(((ProductStatuses)0).GetTitle());
    Console.WriteLine(((ProductStatuses)1).GetTitle());
    Console.WriteLine(((ProductStatuses)3).GetTitle());

    var query_01 = (from item in SH_Products
                    select new
                    {
                        Status = item.Status,
                        StatusText = ((ProductStatuses)item.Status).GetTitle(),
                    });

    var query_02 = (from item in SH_Products.ToList()
                    select new
                    {
                        Status = item.Status,
                        StatusText = ((ProductStatuses)item.Status).GetTitle(),
                    });

    query_01.Dump();
    query_02.Dump();
}

public static class ExtensionMethods
{
    public static string GetTitle(this Enum enm)
    {
        if (enm == null)
        {
            return "Enum is null";
        }

        var type = enm.GetType();
        var members = type.GetMembers(BindingFlags.NonPublic | BindingFlags.Public | BindingFlags.Static);

        foreach (var item in members)
        {
            if (enm.ToString() == item.Name)
            {
                var attributes = item.GetCustomAttributes(typeof(FaraDisplay), false);

                var attribute = String.Empty;
                if (attributes.Length == 0)
                    attribute = item.Name;
                else
                    attribute = ((FaraDisplay)attributes.Single()).GetName();

                return attribute;
            }
        }

        throw new InvalidEnumArgumentException();
    }
}

public enum ProductStatuses
{
    [FaraDisplay("Inactive Product")]
    Inactive = 0,
    [FaraDisplay("Available Product")]
    Available = 1,
    [FaraDisplay("ComingSoon Product")]
    ComingSoon = 2,
    [FaraDisplay("UnAvailable Product")]
    UnAvailable = 3,
    [FaraDisplay("OrderRegister Product")]
    OrderRegister = 4
}

[AttributeUsage(AttributeTargets.Field, AllowMultiple = false)]
public class FaraDisplay : Attribute
{
    private readonly string _name;

    public FaraDisplay(string name)
    {
        _name = name;
    }

    public string GetName()
    {
        return _name;
    }
}
Run Code Online (Sandbox Code Playgroud)

输出:

非活动产品
可用产品
UnAvailable产品

枚举为空
Enum为null
Enum为null
Enum为null

非活动产品
可用产品
不可用产品
不可用产品

Dar*_*opp 8

实际上问题是linq to sql不知道如何转换GetTiltle()成sql语句,所以它会跳过它.当你打电话时ToList(),你正在使用linq到对象,所以它按预期工作,因为那时它只是简单的'.net.

所有linq提供程序(实体框架,nhibernate等)都有同样的问题,尽管你会看到实体框架抛出一个异常,也许也是nhibernate.

根据你正在做的事情,没有办法绕过它,所以你只需要在内存中有对象就可以进行计算.