我想通过传递委托而不是const字符串来增加可维护性。我所做的就是这样;
var propertyName = SprintMetrics.GetNameOf(metric => metric.Productivity); //Should be : "Productivity"
和:
public static string GetNameOf(Func<SprintMetrics, double> valueFunc)
{
return valueFunc.GetMethodInfo().Name; //Result is : <Excute>b_40....
}
在调试期间,我走了“valueFunc”,并且在任何地方都没有“生产力”。
有没有办法让房产名称“生产力”?谢谢。
根据Access Denied的回答,可以通过以下两种方式完成:
var p = nameof(SprintMetrics.Productivity); //"Productivity"
var metrics = new SprintMetrics();
p = nameof(metrics.Productivity); //"Productivity"
您可以使用为此任务设计的C#关键字nameof:
var propertyName = nameof(metric.Productivity)
有关更多信息,请查看以下article。
至于你的代码,为了从lambda表达式中提取属性名,你可以使用以下方法(在这种情况下不需要输入Func参数):
public static string GetPropertyName<TProperty>(Expression<Func<TProperty>> propertyLambda)
{
MemberExpression member = propertyLambda.Body as MemberExpression;
if (member == null)
throw new ArgumentException(string.Format(
"Expression '{0}' refers to a method, not a property.",
propertyLambda.ToString()));
PropertyInfo propInfo = member.Member as PropertyInfo;
if (propInfo == null)
throw new ArgumentException(string.Format(
"Expression '{0}' refers to a field, not a property.",
propertyLambda.ToString()));
return propInfo.Name;
}
你可以这样称呼它:GetPropertyName(() => metric.Productivity)
我走了“valueFunc”,任何地方都没有“生产力”。
这是因为valueFunc
只是一个匿名函数,它返回Productivity
属性的值,因为这是你定义委托的方式。
如果你想要检查委托,那么使用Expression
代替:
public static string GetNameOf<T>(Expression<Func<SprintMetrics, T>> valueFunc)
{
var expression = (MemberExpression)valueFunc.Body;
return expression.Member.Name;
}
当然,你会想要添加错误处理(如果action.Body
不是MemberExpression
怎么办?如果它指的是字段而不是属性怎么办?)。您可以在this answer中看到更完整的示例