Lazy

Lazy<T>延遲加載,主要表達的思想是,把對象的創建將會延遲到使用時創建,而不是在對象實例化時創建對象。這種方式有助於提高於應用程序的性能,避免浪費計算,節省內存的使用等。例如:

class Program
{
    static void Main(string[] args)
    {
        Lazy<Large> lazyObject = new Lazy<Large>();
        Console.WriteLine(lazyObject.IsValueCreated);
        lazyObject.Value.Test();
        Console.WriteLine(lazyObject.IsValueCreated);
    }
}

class Large
{
    public Large() { }
    public void Test() 
    {
        Console.WriteLine("Test");
    }
}
在.NET Framework 4之前,也會有這樣的需求,我們不妨自己實現一個簡單的Lazy:

class MyLazy<T> where T : new()
{
    private T value;
    private bool isLoaded;
    public MyLazy()
    {
        isLoaded = false;
    }
    public T Value
    {
        get
        {
            if (!isLoaded)
            {
                value = new T();
                isLoaded = true;
            }
            return value;
        }
    }
}

這應該是最簡單版本的Lazy<T>了,沒有線程安全檢測,其實什麼都沒有,只有着訪問時創建真實對象,可是對於我們一般的應用來說也許就已經足夠了。實際上.NET Framework和上面的實現原理是一致的。

添加委託參數的Lazy:

class Program
{
    static void Main(string[] args)
    {            
        ///這行代碼表明:要創建一個延遲加載的字符串對象s
        ///原型爲Lazy<T> 對象名=new Lazy<T>(Fun<T>)
        ///採用泛型委託進行構造,實例化此委託時要求必須是返回值T類型的方法
        ///如在本例中,T爲string,則TestLazy.GetString方法的返回值必須也是string類型
        Lazy<string> s = new Lazy<string>(TestLazy.GetString);
        Console.WriteLine(s.IsValueCreated);//返回False
        Console.WriteLine(s.Value); //返回S的當前值
        Console.WriteLine(s.IsValueCreated);//返回True   
     }
}
public class TestLazy
{ 
    public static string GetString()
    {
        return DateTime.Now.ToLongTimeString();
    }
}

下面再用一個例子,演示延遲加載。
  在這個例子中,使用了BlogUser對象,該對象包含多個Article對象,當加載BlogUser對象時,Article對象並不加載,當需要使用Article對象時,才加載。

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;
    }
}

運行的結果爲:



最後說一下,延遲加載主要應用場景:
  當創建一個對象的子對象開銷比較大時,而且有可能在程序中用不到這個子對象,那麼可以考慮用延遲加載的方式來創建子對象。另外一種情況就是當程序一啓動時,需要創建多個對象,但僅有幾個對象需要立即使用,這樣就可以將一些不必要的初始化工作延遲到使用時,這樣可以非常有效的提高程序的啓動速度。比如對於MEF框架中經常會導入部件但是不出石化,例如:
[Import(typeof(User))]
public Lazy<Users.IMetadata> Users;

  這種技術在ORM框架得到了廣泛應用,也並非C#獨有的,比如Java裏的Hibernate框架也使用了這一技術。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章