我有一个自定义属性,用于存储枚举项的显示名称。我编写了一个返回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;
}
}
输出:
非活动产品
可用产品
不可用的产品
枚举为空 枚举为空 枚举为空 枚举为空
非活动产品
可用产品
不可用的产品
不可用的产品
答案 0 :(得分:8)
有效的问题是linq to sql不知道如何将GetTiltle()
翻译成sql语句,所以它会跳过它。当你致电ToList()
时,你正在使用linq来对象,所以它按预期工作,因为它只是简单的'那时.net。
所有linq提供程序(实体框架,nhibernate等)都有同样的问题,尽管你会看到实体框架抛出一个异常,也许也是nhibernate。
根据你正在做的事情,没有办法绕过它,所以你只需要在内存中有对象就可以进行计算。
答案 1 :(得分:0)
在Linq查询中你不应该使用方法调用,而应该首先获取值,然后在linq查询中分配我认为Darren是正确的。