日期:2010-01-31 浏览次数:20496 次
在软件开发中,术语延迟指的是尽可能久地推迟特定的高开销活动的空闲时间。 软件延迟过程中其实也在进行操作,但意味着任何操作仅当需要完成某一特定任务时才会发生。 就这一点而言,延迟是软件开发中的一种重要模式,可以成功地应用于包括设计与实施在内的各种情景中。
例如,极限编程方法中的一种基本编码实践就被简单地概括为“您不会需要它”,那就是一种明确的延迟要求 - 当且仅当您需要这些功能时,才需要在基本代码中包含它。
从另一个角度来看,在实施类的过程中,当要从难以访问的源中加载数据时,您也可能需要延迟。 事实上,延迟加载模式解释了这种普遍接受的解决方案,即定义一个类成员,但使其保持为空,直到其他某些客户端代码实际需要其内容时为止。 延迟加载完全适合在对象关系映射 (ORM) 工具(如实体框架和 NHibernate)环境中使用。 ORM 工具用于映射面向对象的环境与关系数据库之间的数据结构。 例如,在这种环境中,延迟加载指的就是仅当某些代码尝试读取 Customer 类上公开的 Orders 集合属性时,
框架才能加载 Customer 的 Orders。
但是,延迟加载并不限于特定的实施方案(如 ORM 编程)。 而且,延迟加载指的就是在某些数据实际可用之前不获取该数据的实例。 换言之,延迟加载就是要有特殊工厂逻辑,即跟踪必须要创建的内容,最后在实际请求该内容时以静默方式创建该内容。
在 Microsoft .NET Framework 中,开发人员早就在我们的类中手动实施了所有延迟行为。 在 .NET Framework 4 问世之前,从未有过内置的机制来帮助完成此任务。 在 .NET Framework 4 中,我们可以开始
使用全新的 Lazy<T> 类。
了解 Lazy<T> 类
Lazy<T> 是一个特殊的工厂,您可以用来包装给定 T 类型的对象。 Lazy<T> 包装代表一个尚不存在的类实例的实时代理。 使用 Lazy 包装的理由有很多,其中最重要的莫过于可以提高性能。 延迟初始化对象可以避免所有不必要的计算,从而减少内存消耗。 如果加以合理利用,延迟初始化对象也可以成为一种加快应用程序启动的强大工具。 以下代码说明了以延迟方式初始化对象的方法:
var container = new Lazy<DataContainer>();
在本例中,DataContainer 类表示的是一个引用了其他对象数组的纯数据容器对象。 在刚刚对 Lazy<T> 实例调用完 new 运算符之后,返回的只是一个实时的 Lazy<T> 类实例;无论如何都不会得到指定类型 T 的实例。 如果您需要向其他类的成员传递一个 DataContainer 实例,则必须更改这些成员的签名才能使用 Lazy<DataContainer>,如下所示:
void ProcessData(Lazy<DataContainer> container);
何时创建 DataContainer 的实际实例,以便程序可以处理其所需的数据? 让我们来看看 Lazy<T> 类的公共编程接口。 该公共接口非常小,因为它只包含两个属性:Value 和 IsValueCreated。 如果存在与 Lazy 类型关联的实例,则属性 Value 就会返回该实例的当前值。 该属性的定义如下:
public T Value
{
get { ... }
}
属性 IsValueCreated 可以返回一个 Boolean 值,表示 Lazy 类型是否已经过实例化。 以下是该属性的源代码中的一段摘录:
public bool IsValueCreated
{
get
{
return ((m_boxed != null) && (m_boxed is Boxed<T>));
}
}
如果 Lazy<T> 类包含 T 类型的实际实例(如果有),则 m_boxed 成员就是该类的一个内部私有的不稳定成员。 因此,IsValueCreated 只需检查是否存在 T 的实时实例,然后返回一个 Boolean 答案。 如前文所述,m_boxed 成员是私有的并且不稳定(如以下代码段所示):
private volatile object m_boxed;
在 C# 中,volatile 关键字表示成员可以被并发运行的线程修改。 volatile 关键字用于下面这样的成员:这类成员可以在多线程环境中使用,但无法防止多个可能的并发线程同时对其进行访问(本意是出于性能因素考虑)。 我们稍后再回到 Lazy<T> 的线程方面上来。 目前,可以肯定地说,默认情况下 Lazy<T> 的公共成员和受保护成员是线程安全的。 当有任意代码首次尝试访问 Value 成员时,就会创建类型 T 的实际实例。 对象创建方面的详细信息取决于各种线程属性,这些属性可以通过 Lazy<T> 构造函数来指定。 应该明确的是,线程模式的含义仅当 boxed 值实际上已初始化或首次被访问时才很重要。
默认情况下,类型 T 的实例是通过调用 Activator.CreateInstance 进行反射获取的。 以下是一个典型的与 Lazy<T> 类型进行交互的简单示例:
var temp = new Lazy<DataContainer>();
Console.WriteLine(temp.IsValueCreated);
Console.WriteLine(temp.Value.SomeValue);
请注意,在调用 Value 之前,并不一定要对 IsValueCreated 进行检查。 通常情况下,仅当(无论出于何种原因)您需要了解某个值当前是否与 Lazy 类型关联时,才必须查看 IsValueCreated 的值。 您无需检查 IsValueCreated 即可避免发生对 Value 的空引用异常。 以下代码即可保证正常运行:
var temp = new Lazy<DataContainer>();
Console.WriteLine(temp.Value.SomeValue);
Value 属性的 getter 会检查 boxed 值是否已经存在;如果不存在,则会触发逻辑创建一个包装类型实例,并返回该实例。
实例化过程
当然,当该 Lazy 类型(上例中的 DataContainer)的构造函数引发异常时,您的代码会负责处理该异常。 所捕获异常属于 TargetInvocationException 类型,该异常是 .NET 反射无法间接创建某类型实例时收到的典型异常。
Lazy<T> 包装逻辑只能确定是否已创建类型 T 的实例,并不能保证您在访问 T 上的任意公共成员时都不会收到空引用异常。 以下面的代码段为例:
public class DataContainer
{
public DataContainer()
{
}
public IList<String> SomeValues { get; set; }
}
现在假设您尝试从客户端程序调用以下代码:
var temp = new Lazy<DataContainer>();
Console.WriteLine(temp.Value.SomeValues.Count);
在这种情况下,您将收到一个异常,这是因为 DataContainer 对象的 SomeValues 属性为空,而非 DataContainer 本身为空。 引发该异常是因为 DataContainer 的构造函数没有正常初始化其所有成员;该错误与 lazy 方法的实施无关。
Lazy<T> 的 Value 属性为只读属性,即一旦经过初始化,Lazy<T> 对象将始终返回同一个类型 T 实例或同一个值(当 T 为值类型时)。 您无法修改实例,但可以访问该实例可能拥有的所有公共属性。
以下是配置 Lazy<T> 对象向 T 类型传递临时参数的方法:
temp = new Lazy<DataContainer>(() => new Orders(10));
其中一个 Lazy<T> 构造函数会接受一个委托,您可以通过该委托指定为 T 构造函数产生正确输入数据所需的任何操作。 在首