完全限定的属性名称

Jam*_*mes 1 c# nullreferenceexception

我试图防止 System.NullReferenceException。

我有一个公司,它有一个雇员的集合。每个员工都有一组技能。

SelectedEmployee 指向 Employee 集合中当前选定的元素。

SelectedSkill 指向技能集合中当前选定的元素。

我有一个 ListView,它的 ItemSource 绑定到 Skills 集合;ListView 的 SelectedItem 绑定到 SelectedSkill。

删除技能时,我希望 ListView 滚动到最后一个元素。

private void DeleteSelectedSkillFromSelectedEmployee()
{
    Company.SelectedEmployee.Skills.Remove(Company.SelectedEmployee.SelectedSkill);
    EmployeeSkillsListView.ScrollIntoView(Company.SelectedEmployee.Skills.Last());
}
Run Code Online (Sandbox Code Playgroud)

如果尚未选择 Employee,则 SelectedEmployee 将为 null。在方法内部执行任何操作时,这将导致 System.NullReferenceException。

注意:我使用了一个扩展方法来替换 .Last() 所以它不会在空集合上出错。

为了解决这个问题,我使用了一个 Util 方法:

public static class Utils
{
    public static bool PropertyExists(Object obj, String name)
    {
        foreach (String part in name.Split('.'))
        {
            if (obj == null) { return false; }

            Type type = obj.GetType();
            System.Reflection.PropertyInfo info = type.GetProperty(part);

            if (info == null) { return false; }

            obj = info.GetValue(obj, null);
        }
        return obj != null;
    }
}
Run Code Online (Sandbox Code Playgroud)

所以它现在看起来像这样:

private void DeleteSelectedSkillFromSelectedEmployee()
{
    if(Utils.PropertyExists(Company, "SelectedEmployee.SelectedSkill"))
    {
         Company.SelectedEmployee.Skills.Remove(Company.SelectedEmployee.SelectedSkill);
         EmployeeSkillsListView.ScrollIntoView(Company.SelectedEmployee.Skills.Last());
    }
}
Run Code Online (Sandbox Code Playgroud)

以上一切正常。这不是确切的场景或代码,所以不要担心更正上面的任何内容(假设它工作正常)。这只是下面的问题,我真的很想知道。

(想象一下 SelectedEmployee 和 SelectedSkill 不为空)

有没有办法获得属性的完全限定名称?所以我可以做这样的事情:

if(Utils.PropertyExists(Company, GetFullyQualifiedName(Company.SelectedEmployee.SelectedSkill)))
Run Code Online (Sandbox Code Playgroud)

其中 GetFullyQualifiedName(Object) 返回“Company.SelectedEmployee.SelectedSkill”。

问题的第二部分:假设 SelectedEmployee 为 null:是否有任何方法允许将 NullReference 传递给方法?我 99.9% 确定答案是否定的 :)

Jcl*_*Jcl 5

我不明白。为什么不只是:

private void DeleteSelectedSkillFromSelectedEmployee()
{
  if(Company != null &&
     Company.SelectedEmployee != null && 
     Company.SelectedEmployee.Skills != null)
  {           
    Company.SelectedEmployee.Skills.Remove(Company.SelectedEmployee.SelectedSkill);
    EmployeeSkillsListView.ScrollIntoView(Company.SelectedEmployee.Skills.Last());
  }
}
Run Code Online (Sandbox Code Playgroud)

或者在 C#6 中

if(Company?.SelectedEmployee?.Skills != null) 
{
  ...
}
Run Code Online (Sandbox Code Playgroud)

如果您仍然想要使用该GetFullyQualifiedName方法,那么您可以使用的最接近的方法可能是(不检查错误,这只是一个快速的技巧):

public static string GetPathOfProperty<T>(Expression<Func<T>> property)
{
  string resultingString = string.Empty;
  var  p = property.Body as MemberExpression;
  while (p != null)
  {             
    resultingString = p.Member.Name + (resultingString != string.Empty ? "." : "") + resultingString;
    p = p.Expression as MemberExpression;
  }
  return resultingString;           
}
Run Code Online (Sandbox Code Playgroud)

然后像这样使用它:

GetPathOfProperty(() => Foo.Bar.Baz.SomeProperty );
Run Code Online (Sandbox Code Playgroud)

这将返回一个包含 "Foo.Bar.Baz.SomeProperty"

在小提琴中检查它