延迟初始化就是将对象的初始化延迟到第一次使用该对象时。延迟初始化是我们在写程序时经常会遇到的情形,例如创建某一对象时需要花费很大的开销,而这一对象在系统的运行过程中不一定会用到,这时就可以使用延迟初始化,在第一次使用该对象时再对其进行初始化,使用延迟初始化可以提高程序的效率,使程序占用更少的内存。在 .NET 4.0 之前要实现延迟初始化,需要我们自己动手编写具体的实现方式(关于延迟初始化的实现方式,可以参看这个 http://msdn.microsoft.com/en-us/vcsharp/bb870976.aspx .NET 4.0 的 Lazy<T> 的实现方式与此相似)。在 .NET 4.0 中提供了一个泛型类 System.Lazy<T> 可以帮助我们实现延迟初始化。
首先我们来看下面的代码:
03 |
static void Main( string [] args) |
05 |
Lazy<Customer> customer = new Lazy<Customer>(); |
06 |
if (!customer.IsValueCreated) |
07 |
Console.WriteLine( "Customer 尚未初始化。" ); |
09 |
Console.WriteLine( "客户姓名是:{0}" , customer.Value.Name); |
11 |
if (customer.IsValueCreated) |
12 |
Console.WriteLine( "Customer 已经初始化。" ); |
23 |
Console.WriteLine( "调用 Customer 的构造函数。" ); |
26 |
public string Name { get ; set ; } |
28 |
public List<Order> Orders { get ; set ; } |
33 |
public string ID { get ; set ; } |
下面是输出结果:
Customer 尚未初始化。
调用 Customer 的构造函数。
客户姓名是:张三
Customer 已经初始化。
当 customer.Value 被调用时 Customer 才被初始化。只读属性 Value 返回延迟初始化的对象,布尔属性 IsValueCreated 标识延迟初始化的对象是否已经初始化。在上面的示例代码中是在 Customer 类的构造函数中设置的 Name 属性的值,我们可以使用 Lazy<T> 的重载函数 Lazy< T> (Func< T> ) 传入一个带返回值的委托来设置延迟初始化对象的属性值。
1 |
Lazy<Customer> customer1 = new Lazy<Customer>( |
4 |
return new Customer { Name = "李四" }; |
以下是输出结果:
Customer 尚未初始化。
调用 Customer 的构造函数。
客户姓名是:李四
Customer 已经初始化。
可以看到 customer1 的 Value 属性使用的是通过委托传入的参数初始化的。
要实现公共属性的延迟初始化,将属性对应的字段定义为 Lazy<T>,然后通过 get 访问器返回字段的 Value 属性。
06 |
Console.WriteLine( "调用 Customer 的构造函数。" ); |
08 |
_orders = new Lazy<List<Order>>(() => |
11 |
return new List<Order> |
13 |
new Order { ID = "1" }, |
14 |
new Order { ID = "2" }, |
15 |
new Order { ID = "3" } |
21 |
public string Name { get ; set ; } |
23 |
private Lazy<List<Order>> _orders; |
24 |
public List<Order> Orders |
35 |
public string ID { get ; set ; } |
Lazy<T> 对象初始化默认是线程安全的,在多线程环境下,第一个访问 Lazy<T> 对象的 Value 属性的线程将初始化 Lazy<T> 对象,以后访问的线程都将使用第一次初始化的数据。
参考:http://msdn.microsoft.com/en-us/library/dd997286.aspx