0

如果在路径中的任何地方遇到空引用,是否有可以在链中调用的 DynamicObject 实现保持空引用结束,而不抛出任何异常?

a.b.c.e

例如:如果 a 为空则 abce 为空,或者如果 c 为空 ce 为空等?

很像 Haskell 的 Maybe monad。

4

3 回答 3

1

你可以做类似的事情,但不能针对最外层的对象,即如果a为空,你不能访问a.b.

您可以创建A该类的空实例,该实例返回其所有属性的空实例。然后a.b将返回 的空实例B,对于该c属性将返回一个空实例C,对于该e属性将返回一个空实例E

你不会得到一个空值,但你会得到一个空实例,你可以检查:

E e = a.b.c.e;
if (e != E.Empty) { ... }

如果沿途的任何属性返回一个空实例,则最终结果将是E.Empty.

public class A {

  public B b;

  public A(B newB) { b = newB; }

  private static A _empty = new A(B.Empty);
  public static A Empty { get { return _empty; }}

}

public class B {

  public C c;

  public B(C newC) { c = newC; }

  private static B _empty = new B(C.Empty);
  public static B Empty { get { return _empty; } }

}

public class C {

  public E e;

  public C(E newE) { e = newE; }

  private static C _empty = new C(E.Empty);
  public static C Empty { get { return _empty; } }

}

public class E {

  public string name;

  public E(string newName) { name = newName; }

  private static E _empty = new E(null);
  public static E Empty { get { return _empty; } }

}

例子:

A a1 = new A(new B(new C(new E("Hello world!"))));
A a2 = new A(new B(new C(E.Empty)));
A a3 = new A(B.Empty);

E e1 = a1.b.c.e; // e1.name returns "Hello world!"
E e2 = a2.b.c.e; // e2 == E.Empty
E e3 = a3.b.c.e; // e3 == E.Empty
于 2010-10-22T18:13:53.057 回答
1

查看这篇很棒的文章:Chained null checks and the Maybe monad

很多程序员都遇到过这样一种情况,即在访问嵌套对象属性(例如,person.Address.PostCode)时,他们必须进行多次空值检查。这个要求经常出现在 XML 解析中,当您尝试访问它们时,缺少的元素和属性可能会返回 null(随后尝试访问 Value 会引发 NullReferenceException)。在本文中,我将展示如何使用 C# 中的 Maybe monad 以及使用扩展方法来提高可读性。

于 2010-10-23T01:25:01.083 回答
1

这是一个穷人的安全导航扩展方法,它只是将一个表达式包装在一个 try catch 中以寻找一个 nullref。

https://gist.github.com/1030887

public static class Extensions
{
    public static TResult SafeInvoke<TModel, TResult>(this TModel model, Func<TModel, TResult> expression, TResult nullValue = default(TResult))
    {
        try
        {
            return expression(model);
        }
        catch (NullReferenceException)
        {
            return nullValue;
        }
    }
}

您可以相当容易地调用代码。

public class MyModel
{
  public Name Name { get; set; }
}

public class Name
{
  public string First { get; set; }
  public string Last { get; set; }
}

var model = new MyModel();
var firstName = model.SafeInvoke(x => x.Name.First, "john");
var lastName = model.SafeInvoke(x => x.Name.Last, "doe");

Console.WriteLine("{0}, {1}", lastName, firstName)
// prints: "doe, john"
于 2011-08-14T04:43:51.350 回答