我有一个问题困扰了我一段时间,但是我找不到答案。
我需要获取Lambda表达式中所引用的属性名称。我会将Lambda表达式提供给一个方法,然后该方法会返回一个字符串。例如,如果我有以下代码:
x => x.WeirdPropertyName
那么该方法将返回:
"WeirdPropertyName"
我已经阅读了可以使用表达式树来完成,但是这个答案让我难以理解。
感谢任何帮助。
我有一个问题困扰了我一段时间,但是我找不到答案。
我需要获取Lambda表达式中所引用的属性名称。我会将Lambda表达式提供给一个方法,然后该方法会返回一个字符串。例如,如果我有以下代码:
x => x.WeirdPropertyName
那么该方法将返回:
"WeirdPropertyName"
我已经阅读了可以使用表达式树来完成,但是这个答案让我难以理解。
感谢任何帮助。
请看这里:
string GetPropertyName<T>(Expression<Func<T>> property)
{
var propertyInfo = (property.Body as MemberExpression).Member as PropertyInfo;
if (propertyInfo == null)
{
throw new ArgumentException("The lambda expression 'property' should point to a valid Property");
}
return propertyInfo.Name;
}
x => x.WeirdPropertyName
等表达式不同,它还可以处理“扩展”表达式,如x => x.WeirdMember.WeirdPropertyName
。
以下是该答案中的代码:
// code adjusted to prevent horizontal overflow
static string GetFullPropertyName<T, TProperty>
(Expression<Func<T, TProperty>> exp)
{
MemberExpression memberExp;
if (!TryFindMemberExpression(exp.Body, out memberExp))
return string.Empty;
var memberNames = new Stack<string>();
do
{
memberNames.Push(memberExp.Member.Name);
}
while (TryFindMemberExpression(memberExp.Expression, out memberExp));
return string.Join(".", memberNames.ToArray());
}
// code adjusted to prevent horizontal overflow
private static bool TryFindMemberExpression
(Expression exp, out MemberExpression memberExp)
{
memberExp = exp as MemberExpression;
if (memberExp != null)
{
// heyo! that was easy enough
return true;
}
// if the compiler created an automatic conversion,
// it'll look something like...
// obj => Convert(obj.Property) [e.g., int -> object]
// OR:
// obj => ConvertChecked(obj.Property) [e.g., int -> long]
// ...which are the cases checked in IsConversion
if (IsConversion(exp) && exp is UnaryExpression)
{
memberExp = ((UnaryExpression)exp).Operand as MemberExpression;
if (memberExp != null)
{
return true;
}
}
return false;
}
private static bool IsConversion(Expression exp)
{
return (
exp.NodeType == ExpressionType.Convert ||
exp.NodeType == ExpressionType.ConvertChecked
);
}