淺析C#延遲加載的運(yùn)行機(jī)制和應(yīng)用場(chǎng)景
延遲加載,亦稱(chēng)延遲實(shí)例化,延遲初始化等,主要表達(dá)的思想是,把對(duì)象的創(chuàng)建將會(huì)延遲到使用時(shí)創(chuàng)建,而不是在對(duì)象實(shí)例化時(shí)創(chuàng)建對(duì)象,即用時(shí)才加載。這種方式有助于提高于應(yīng)用程序的性能,避免浪費(fèi)計(jì)算,節(jié)省內(nèi)存的使用等。針對(duì)于這種做法,似乎稱(chēng)之為即用即創(chuàng)建更為合適些。
先來(lái)看一下在Framework4.0中如何實(shí)現(xiàn)延遲加載。
Framework4.0提供了一個(gè)包裝類(lèi) Lazy<T>,可以輕松的實(shí)現(xiàn)延遲加載。
- ///這行代碼表明:要?jiǎng)?chuàng)建一個(gè)延遲加載的字符串對(duì)象s
- ///原型為L(zhǎng)azy<T> 對(duì)象名=new Lazy<T>(Fun<T>)
- ///采用泛型委托進(jìn)行構(gòu)造,實(shí)例化此委托時(shí)要求必須是返回值T類(lèi)型的方法
- ///如在本例中,T為string,則TestLazy.GetString方法的返回值必須也是string類(lèi)型
- Lazy<string> s = new Lazy<string>(TestLazy.GetString);
TestLazy.GetString()方法如下示:
- public class TestLazy {
- public static string GetString()
- {
- return DateTime.Now.ToLongTimeString();
- } }
可以通過(guò)IsValueCreated屬性來(lái)確定對(duì)象是否已創(chuàng)建,通過(guò)Value屬性來(lái)獲取當(dāng)前對(duì)象的值。
- Console.WriteLine(s.IsValueCreated);//返回False
- Console.WriteLine(s.Value); //返回s的當(dāng)前值
- Console.WriteLine(s.IsValueCreated);//返回True
下面經(jīng)出完整代碼,以供測(cè)試
字符串對(duì)象延遲加載完整示例
- class Program
- {
- static void Main(string[] args)
- {
- ///這行代碼表明:要?jiǎng)?chuàng)建一個(gè)延遲加載的字符串對(duì)象s
- ///原型為L(zhǎng)azy<T> 對(duì)象名=new Lazy<T>(Fun<T>)
- ///采用泛型委托進(jìn)行構(gòu)造,實(shí)例化此委托時(shí)要求必須是返回值T類(lèi)型的方法
- ///如在本例中,T為string,則TestLazy.GetString方法的返回值必須也是string類(lèi)型
- Lazy<string> s = new Lazy<string>(TestLazy.GetString);
- Console.WriteLine(s.IsValueCreated);//返回False
- Console.WriteLine(s.Value); //返回S的當(dāng)前值
- Console.WriteLine(s.IsValueCreated);//返回True
- }
- }
- public class TestLazy
- {
- public static string GetString()
- {
- return DateTime.Now.ToLongTimeString();
- }
- }
下面再用一個(gè)例子,演示延遲加載。
在這個(gè)例子中,使用了BlogUser對(duì)象,該對(duì)象包含多個(gè)Article對(duì)象,當(dāng)加載BlogUser對(duì)象時(shí),Article對(duì)象并不加載,當(dāng)需要使用Article對(duì)象時(shí),才加載。
包含子對(duì)象的延遲加載
- class Program
- {
- static void Main(string[] args)
- {
- BlogUser blogUser = new BlogUser(1);
- Console.WriteLine("blogUser has been initialized");
- foreach (var article in blogUser.Articles.Value)
- {
- Console.WriteLine(article.Title);
- }
- }
- }
- public class BlogUser
- {
- public int Id { get; private set; }
- public Lazy<List<Article>> Articles { get; private set; }
- public BlogUser(int id)
- {
- this.Id = id;
- Articles =new Lazy<List<Article>>(()=>ArticleServices.GetArticesByID(id));
- Console.WriteLine("BlogUser Initializer");
- }
- }
- public class Article
- {
- public int Id { get; set; }
- public string Title{get;set;}
- public DateTime PublishDate { get; set; }
- }
- public class ArticleServices
- {
- public static List<Article> GetArticesByID(int blogUserID)
- {
- List<Article> articles = new List<Article> {
- new Article{Id=1,Title="Lazy Load",PublishDate=DateTime.Parse("2011-4-20")},
- new Article{Id=2,Title="Delegate",PublishDate=DateTime.Parse("2011-4-21")},
- new Article{Id=3,Title="Event",PublishDate=DateTime.Parse("2011-4-22")},
- new Article{Id=4,Title="Thread",PublishDate=DateTime.Parse("2011-4-23")}
- };
- Console.WriteLine("Article Initalizer");
- return articles;
- }
- }
運(yùn)行結(jié)果如圖示:
最后說(shuō)一下,延遲加載主要應(yīng)用場(chǎng)景
當(dāng)創(chuàng)建一個(gè)對(duì)象的子對(duì)象開(kāi)銷(xiāo)比較大時(shí),而且有可能在程序中用不到這個(gè)子對(duì)象,那么可以考慮用延遲加載的方式來(lái)創(chuàng)建子對(duì)象。另外一種情況就是當(dāng)程序一啟動(dòng)時(shí),需要?jiǎng)?chuàng)建多個(gè)對(duì)象,但僅有幾個(gè)對(duì)象需要立即使用,這樣就可以將一些不必要的初始化工作延遲到使用時(shí),這樣可以非常有效的提高程序的啟動(dòng)速度。
這種技術(shù)在ORM框架得到了廣泛應(yīng)用,也并非C#獨(dú)有的,比如Java里的Hibernate框架也使用了這一技術(shù)。
原文鏈接:http://www.cnblogs.com/yangyancheng/archive/2011/04/26/2029209.html






