父级空值的可变分辨率

我有一个方法,其中包含很多这样的语句:

employee.LastName = file.EMPLOYEE.NAME.SUBNAME.FAMILYNAME.NAMEPART1.Value;

(这是一个组合结构.实际结构基于2.3 HL7 XML,并且复杂得多,并且充满了毫无意义的对象.)

所以我的问题是这些值中的任何一个都可以为null.如果我不进行检查,那么当我尝试获取链中的下一个值时,它将引发异常.

显然我可以做这样的事情:

if (contract.EMPLOYEE != null)
 if (contract.EMPLOYEE.NAME.SUBNAME != null)
  if (contract.EMPLOYEE.NAME.FAMILYNAME != null)
   if (contract.EMPLOYEE.NAME.FAMILYNAME.NAMEPART1 != null)
     employee.LastName = file.EMPLOYEE.NAME.SUBNAME.FAMILYNAME.NAMEPART1.Value;  

但这每次我需要设置一个值时都要进行很多检查. (对象树发生了很大变化,因此我无法对其进行一次检查并无法完成系统.)

也许更好的方法是这样做:

try
{
   employee.LastName = file.EMPLOYEE.NAME.SUBNAME.FAMILYNAME.NAMEPART1.Value;
}
catch
{}

(请注意,如果该值不存在,那么我什么也没做就可以了.)

但是,这仍然有些混乱.我仅用此方法就可以设置14个项目,并且还有很多其他项目.

有什么办法可以使我抽象化,这样我就可以做我的二传手而已?

我想到了可以执行check / try语句的方法调用,但是如果我传入file.EMPLOYEE.NAME.SUBNAME.FAMILYNAME.NAMEPART1.Value,它将在我通过它时尝试解决它(因此给了我一个例外) .

还有其他(干净的)方法吗? (也许可以使用扩展方法?)我将不胜感激.

注意:我最终只是对每一行都这样做:

try { employee.LastName = file.EMPLOYEE.NAME.SUBNAME.FAMILYNAME.NAMEPART1.Value; } catch (System.NullReferenceException) { }

它似乎比lambda系统更具可读性.

解决方法:

这个怎么样:

static class Coalesce
{
    static public TResult C<T, TResult>(this T obj, Func<T, TResult> func)
        where TResult : class
    {
        if (obj != null)
            return func(obj);
        return null;
    }
}

像这样使用:

employee.LastName =
    file.C(o => o.EMPLOYEE)
        .C(o => o.NAME)
        .C(o => o.SUBNAME)
        .C(o => o.FAMILYNAME)
        .C(o => o.NAMEPART1)
        .C(o => o.Value); 

如果链中的任何属性为null,则将返回null.

这是由问题Extending the C# Coalesce Operator启发的,但是我使用了稍微不同的方法.以稍微更长的调用语法为代价,我的扩展类更加简单,并且链中的链接数没有限制.

您也可以将其与??结合使用运算符以分配默认值:

class Foo { public Bar MyBar { get; } }
class Bar { public Baz MyBaz { get; } }
class Baz { }

Baz baz = foo.C(o => o.MyBar).C(o => o.MyBaz) ?? new Baz();
上一篇:javascript-JS:重命名变量以进行重构(使用AST,而不是文本)


下一篇:我可以设置ReSharper来将未导入的类用法重命名为其全名,而不是导入其名称空间吗?