访问类的成员的类型安全枚举

时间:2013-10-03 21:32:40

标签: c#

我的班级MyClass<t>有以下方法

private static bool TypeHasProperty(string NameOfPropertyToMatch)
{
    var properties = typeof(t).GetProperties();
    var requiredProperty = properties.First(
            a => a.Name == NameOfPropertyToMatch);
    if (requiredProperty == null)
    {
        return false
    };

    return true;
}

这是在静态方法开始时调用的:

MyClass<t>.SendToJavaScript(t InstanceOfType, string NameOfPropertyWithinType)

在继续之前确保InstanceOfType实际上具有该名称的属性,因为否则直到waaaaay下线才会抛出异常 (此信息最终将被序列化并发送到Javascript应用程序,该应用程序需要知道要访问哪个属性才能完成它的工作。)

我想要一个好的,类型安全的方法来调用那个方法,如果我决定稍后更改我的属性的名称就不会导致验证失败。

例如,我不想这样称呼它:

MyClass<Person>.SendToJavaScript(MyPerson, "Name")

因为如果我决定在Name课程中将LastName,更改为Person,我必须确保进入并更改该魔术字符串。

我想要的是这样的:

MeClass<Person>.SendToJavaScript(
    MyPerson,
    TypeOf(Person).Properties.Name.ToString())

因此,如果我使用重构将Name更改为LastName,IDE将会扫描并更改我的调用。

有这样的事吗?或者我在重构时只需要小心一点? (我爱我一些F2

2 个答案:

答案 0 :(得分:5)

我们在代码中使用这样的东西:

private static bool TypeHasProperty<P>(Expression<Func<T, P>> accessor)
{
    var propertyName = ((MemberExpression)accessor.Body).Member.Name;
    var properties = typeof(T).GetProperties();
    return properties.Any(a => a.Name == propertyName);
}

使用它:

class Foo
{
    public int Baz { get; set; }
}

MyClass<Foo>.TypeHasProperty(f => f.Baz);

修改:更改了查询以使用Any(),因为它更好地表达了您实际想要做的事情(具有给定名称的属性是否存在?)

请注意,如果未找到匹配项,则会First()

答案 1 :(得分:4)

试试这个:

MeClass<Person>.SendToJavaScript(MyPerson, x => x.Name);

然后:

private static bool TypeHasProperty(Expression propertyExpression)
{
    var expression = GetMemberInfo(propertyExpression);
    string name = expression.Member.Name;

    return typeof(t).GetProperty(name) != null;
}

private static MemberExpression GetMemberInfo(Expression method)
{
    LambdaExpression lambda = method as LambdaExpression;
    if (lambda == null)
        throw new ArgumentNullException("method");

    MemberExpression memberExpr = null;

    if (lambda.Body.NodeType == ExpressionType.Convert)
    {
        memberExpr = 
            ((UnaryExpression)lambda.Body).Operand as MemberExpression;
    }
    else if (lambda.Body.NodeType == ExpressionType.MemberAccess)
    {
        memberExpr = lambda.Body as MemberExpression;
    }

    if (memberExpr == null)
        throw new ArgumentException("method");

    return memberExpr;
}

从这里无耻地采取:Retrieving Property name from lambda expression

但这是一个稍微不同的问题,所以我发布了一个完整的答案。

我刚才注意到,在我提供的链接中,第二个答案对此更为简单。 ChrisWue'a在这个帖子中回答了这个问题。