網站首頁 編程語言 正文
".NET泛型"系列:
ASP.NET泛型一之泛型簡介與基本語法
ASP.NET泛型二之泛型的使用方法
ASP.NET泛型三之使用協變和逆變實現類型轉換
ASP.NET泛型四之使用Lazy<T>實現延遲加載
對于一些"大對象"的創建,我們常常希望延遲加載,即在需要的時候再創建對象實例。現在Lazy<T>很好地支持了這一特點。主要包括:
沒有Lazy<T>之前
在沒有Lazy<T>之前,我們通過如下方式實現延遲加載。
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();
}
}
以上
- 通過私有構造函數屏蔽了LazySingleton類通過構造函數創建的方式
- 私有嵌套類Lazy的data字段負責提供一個LazySingleton的實例
- 只能通過LazySingleton的屬性Instance,才能拿到內部類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>的構造函數重載創建對象,再通過體現延遲加載的Value屬性來實現對象的創建,并獲取對象實例。
public class SomeClass
{
public int ID{get;set;}
}
Lazy<SomeClass> temp = new Lazy<SomeClass>();
Console.WriteLine(temp.Value.ID);
以上,只適用于沒有構造函數的情況,如果有構造函數如何處理呢?
--使用public Lazy(Func<T> valueFactory),通過委托創建對象
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);
延遲加載的本質
創建自定義延遲加載類。
public class MyLazy<T>
{
private volatile object boxed; //volatile說明在多線程狀況下,也可以修改該字段
private Func<T> valueFactory; //委托,用來生產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;
}
//創建內部類實例
private Boxed CreateValue()
{
//如果創建對象實例的委托valueFactory存在
if (this.valueFactory != null)
{
//就通過委托生成對象實例
return new Boxed(this.valueFactory());
}
else
{
//否則,通過反射生成對象實例
return new Boxed((T)Activator.CreateInstance(typeof(T)));
}
}
//內部嵌套類,通過構造函數對其字段賦值
private class Boxed
{
internal T value;
internal Boxed(T value)
{
this.value = value;
}
}
}
自定義帶構造函數的類。
public class Big
{
public int ID { get; set; }
public Big(int id)
{
this.ID = id;
}
}
自定義創建對象實例的工廠類。
public class BigFactory
{
public static Big Build()
{
return new Big(10);
}
}
客戶端調用。
class Program
{
static void Main(string[] args)
{
MyLazy<Big> temp = new MyLazy<Big>(() => BigFactory.Build());
Console.WriteLine(temp.Value.ID);
Console.ReadKey();
}
}
延遲加載的本質大致是:
- 由延遲加載類的內部嵌套類產生對象實例
- 再通過延遲加載類的某個屬性來延遲獲取對象實例,而對象實例是通過委托等方式創建的
原文鏈接:https://www.cnblogs.com/darrenji/p/3852626.html
相關推薦
- 2022-05-27 C++?算法精講之貪心算法_C 語言
- 2021-11-25 使用Xshell連接VMware上的Linux虛擬機(圖文步驟)_VMware
- 2023-11-18 Linux中ARM設備Linux 使用iperf3測量網絡帶寬
- 2023-10-12 v-if和v-for的優先級以及二者同時使用的情況
- 2022-05-27 numpy模塊中axis的理解與使用_python
- 2022-01-10 Set、Map、WeakSet 和 WeakMap 的區別?
- 2023-12-22 MAC電腦添加hosts
- 2022-07-08 Python數據分析之使用matplotlib繪制折線圖、柱狀圖和柱線混合圖_python
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支