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

时间:2015-01-31 07:25:05

标签: c# linq enums extension-methods

我有一个自定义属性,用于存储枚举项的显示名称。我编写了一个返回Enum自定义属性显示名称的扩展方法。

我在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;
    }
}

输出:

  

非活动产品
  可用产品
  不可用的产品
  
  枚举为空   枚举为空   枚举为空   枚举为空   
  非活动产品
  可用产品
  不可用的产品
  不可用的产品

2 个答案:

答案 0 :(得分:8)

有效的问题是linq to sql不知道如何将GetTiltle()翻译成sql语句,所以它会跳过它。当你致电ToList()时,你正在使用linq来对象,所以它按预期工作,因为它只是简单的'那时.net。

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

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

答案 1 :(得分:0)

在Linq查询中你不应该使用方法调用,而应该首先获取值,然后在linq查询中分配我认为Darren是正确的。

相关问题