ASP.NET泛型四之使用Lazy<T>實現(xiàn)延遲加載
".NET泛型"系列:
ASP.NET泛型三之使用協(xié)變和逆變實現(xiàn)類型轉(zhuǎn)換
ASP.NET泛型四之使用Lazy<T>實現(xiàn)延遲加載
對于一些"大對象"的創(chuàng)建,我們常常希望延遲加載,即在需要的時候再創(chuàng)建對象實例?,F(xiàn)在Lazy<T>很好地支持了這一特點。主要包括:
沒有Lazy<T>之前
在沒有Lazy<T>之前,我們通過如下方式實現(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(); } }
以上
- 通過私有構(gòu)造函數(shù)屏蔽了LazySingleton類通過構(gòu)造函數(shù)創(chuàng)建的方式
- 私有嵌套類Lazy的data字段負(fù)責(zé)提供一個LazySingleton的實例
- 只能通過LazySingleton的屬性Instance,才能拿到內(nèi)部類Lazy.data所代表的實例
Lazy<T>實例
先看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(); }
通過Lazy<T>的構(gòu)造函數(shù)重載創(chuàng)建對象,再通過體現(xiàn)延遲加載的Value屬性來實現(xiàn)對象的創(chuàng)建,并獲取對象實例。
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),通過委托創(chuàng)建對象
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說明在多線程狀況下,也可以修改該字段 private Func<T> valueFactory; //委托,用來生產(chǎn)T對象實例 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(); } } //初始化對象實例 private T Init() { Boxed boxed = null; if (this.boxed == null) { boxed = this.CreateValue(); this.boxed = boxed; } return boxed.value; } //創(chuàng)建內(nèi)部類實例 private Boxed CreateValue() { //如果創(chuàng)建對象實例的委托valueFactory存在 if (this.valueFactory != null) { //就通過委托生成對象實例 return new Boxed(this.valueFactory()); } else { //否則,通過反射生成對象實例 return new Boxed((T)Activator.CreateInstance(typeof(T))); } } //內(nèi)部嵌套類,通過構(gòu)造函數(shù)對其字段賦值 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)建對象實例的工廠類。
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)生對象實例
- 再通過延遲加載類的某個屬性來延遲獲取對象實例,而對象實例是通過委托等方式創(chuàng)建的
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請查看下面相關(guān)鏈接
相關(guān)文章
深入Lumisoft.NET組件POP3郵件接收與刪除操作的使用詳解
本篇文章對Lumisoft.NET組件POP3郵件接收與刪除操作的使用進(jìn)行了詳細(xì)的介紹。需要的朋友參考下2013-05-05ASP.NET WebForm中<%=%>與<%#%>的區(qū)別
這篇文章主要介紹了ASP.NET WebForm中<%=%>與<%#%>的區(qū)別,需要的朋友可以參考下2015-01-01asp.net下實現(xiàn)URL重寫技術(shù)的代碼
asp.net下實現(xiàn)URL重寫技術(shù)的代碼...2007-10-10asp.net 擴(kuò)展GridView 增加單選按鈕列的代碼
asp.net 擴(kuò)展GridView 增加單選按鈕列的代碼2010-02-02DropDownList綁定選擇數(shù)據(jù)報錯提示異常解決方案
DropDownList控件在綁定選擇數(shù)據(jù)時提示報錯異常詳細(xì)信息為:有一個無效 SelectedValue,因為它不在項目列表中,應(yīng)該有很多新手朋友們遇到過吧,本文將給予解決方法,感興趣的朋友可以了解下,希望本對你有所幫助2013-01-01Asp.net?core?使用SignalR推送消息過程詳解
ASP.NET?Core?SignalR?是一個開放源代碼庫,可用于簡化向應(yīng)用添加實時?Web?功能。?實時?Web?功能使服務(wù)器端代碼能夠?qū)?nèi)容推送到客戶端,本文重點給大家介紹Asp.net?core?使用SignalR推送消息,感興趣的朋友一起看看吧2022-03-03