ASP.NET泛型四之使用Lazy<T>實(shí)現(xiàn)延遲加載
".NET泛型"系列:
對(duì)于一些"大對(duì)象"的創(chuàng)建,我們常常希望延遲加載,即在需要的時(shí)候再創(chuàng)建對(duì)象實(shí)例?,F(xiàn)在Lazy<T>很好地支持了這一特點(diǎn)。主要包括:
沒有Lazy<T>之前
在沒有Lazy<T>之前,我們通過(guò)如下方式實(shí)現(xiàn)延遲加載。
public class LazySinleton
{
private LazySingleton()
{}
public static LazySingleton Instance
{
get
{
return Lazy.data;
}
}
private class Lazy
{
static Lazy()
{}
internal static readonly LazySingleton data = new LazySingleton();
}
}
以上
- 通過(guò)私有構(gòu)造函數(shù)屏蔽了LazySingleton類通過(guò)構(gòu)造函數(shù)創(chuàng)建的方式
- 私有嵌套類Lazy的data字段負(fù)責(zé)提供一個(gè)LazySingleton的實(shí)例
- 只能通過(guò)LazySingleton的屬性Instance,才能拿到內(nèi)部類Lazy.data所代表的實(shí)例
Lazy<T>實(shí)例
先看Lazy<T>的定義:
public class Lazy<T>
{
public Lazy();
public Lazy(bool isThreadSafe);
public Lazy(Func<T> valueFactory);
public Lazy(LazyThreadSafeMode mode);
public Lazy(Func<T> valueFactory, bool isThreadSafe);
public Lazy(Funct<T> valueFactory, LazyThreadSafetyMode mode);
public bool IsValueCreated{get;}
public T Value {get;}
public override string ToStirng();
}
通過(guò)Lazy<T>的構(gòu)造函數(shù)重載創(chuàng)建對(duì)象,再通過(guò)體現(xiàn)延遲加載的Value屬性來(lái)實(shí)現(xiàn)對(duì)象的創(chuàng)建,并獲取對(duì)象實(shí)例。
public class SomeClass
{
public int ID{get;set;}
}
Lazy<SomeClass> temp = new Lazy<SomeClass>();
Console.WriteLine(temp.Value.ID);
以上,只適用于沒有構(gòu)造函數(shù)的情況,如果有構(gòu)造函數(shù)如何處理呢?
--使用public Lazy(Func<T> valueFactory),通過(guò)委托創(chuàng)建對(duì)象
pubic class SomeClass
{
public int ID{get;set;}
public SomeClass(int id)
{
this.ID = id;
}
}
Lazy<SomeClass> temp = new Lazy<SomeClass>(() => new Big(100));
Console.WriteLine(temp.Value.ID);
延遲加載的本質(zhì)
創(chuàng)建自定義延遲加載類。
public class MyLazy<T>
{
private volatile object boxed; //volatile說(shuō)明在多線程狀況下,也可以修改該字段
private Func<T> valueFactory; //委托,用來(lái)生產(chǎn)T對(duì)象實(shí)例
static MyLazy(){}
public MyLazy(){}
public MyLazy(Func<T> valueFactory)
{
this.valueFactory = valueFactory;
}
public T Value
{
get
{
Boxed boxed = null;
if (this.boxed != null)
{
boxed = this.boxed as Boxed;
if (boxed != null)
{
return boxed.value;
}
}
return this.Init();
}
}
//初始化對(duì)象實(shí)例
private T Init()
{
Boxed boxed = null;
if (this.boxed == null)
{
boxed = this.CreateValue();
this.boxed = boxed;
}
return boxed.value;
}
//創(chuàng)建內(nèi)部類實(shí)例
private Boxed CreateValue()
{
//如果創(chuàng)建對(duì)象實(shí)例的委托valueFactory存在
if (this.valueFactory != null)
{
//就通過(guò)委托生成對(duì)象實(shí)例
return new Boxed(this.valueFactory());
}
else
{
//否則,通過(guò)反射生成對(duì)象實(shí)例
return new Boxed((T)Activator.CreateInstance(typeof(T)));
}
}
//內(nèi)部嵌套類,通過(guò)構(gòu)造函數(shù)對(duì)其字段賦值
private class Boxed
{
internal T value;
internal Boxed(T value)
{
this.value = value;
}
}
}
自定義帶構(gòu)造函數(shù)的類。
public class Big
{
public int ID { get; set; }
public Big(int id)
{
this.ID = id;
}
}
自定義創(chuàng)建對(duì)象實(shí)例的工廠類。
public class BigFactory
{
public static Big Build()
{
return new Big(10);
}
}
客戶端調(diào)用。
class Program
{
static void Main(string[] args)
{
MyLazy<Big> temp = new MyLazy<Big>(() => BigFactory.Build());
Console.WriteLine(temp.Value.ID);
Console.ReadKey();
}
}
延遲加載的本質(zhì)大致是:
- 由延遲加載類的內(nèi)部嵌套類產(chǎn)生對(duì)象實(shí)例
- 再通過(guò)延遲加載類的某個(gè)屬性來(lái)延遲獲取對(duì)象實(shí)例,而對(duì)象實(shí)例是通過(guò)委托等方式創(chuàng)建的
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)的支持。如果你想了解更多相關(guān)內(nèi)容請(qǐng)查看下面相關(guān)鏈接
相關(guān)文章:
1. ASP.NET MVC使用正則表達(dá)式驗(yàn)證手機(jī)號(hào)碼2. ASP.NET Core實(shí)現(xiàn)中間件的幾種方式3. ASP.NET Core 依賴注入生命周期示例詳解4. ASP.NET堆和棧四之對(duì)托管和非托管資源的垃圾回收和內(nèi)存分配5. 如何在?ASP.NET?Core?Web?API?中處理?Patch?請(qǐng)求6. ASP.NET Core按用戶等級(jí)授權(quán)的方法7. Asp.Net部署Docker-v指令使用詳解8. ASP.NET泛型二之泛型的使用方法9. ASP.NET MVC實(shí)現(xiàn)下拉框多選10. ASP.NET延遲調(diào)用或多次調(diào)用第三方Web API服務(wù)

網(wǎng)公網(wǎng)安備