延迟初始化 是一种将对象的创建延迟到第一次需要用时的技术,换句话说,对象的初始化是发生在真正需要的时候才执行,值得注意的是,术语 延迟初始化 和 延迟实例化 的意思是相同的——可以互换使用,通过使用 延迟初始化 技术,可以避免应用程序不必要的计算和内存消耗,这篇文章我们将会讨论如何在 C# 中使用 延迟初始化。
有些朋友听完这些可能会懵逼,接下来用一个简单的例子来了解下 延迟加载 的场景,考虑下面两个类, Customer 和 Order , Customer 类包含了一个 Orders 属性,一个人肯定会有很多的订单,也就意味着它可能包含了很多的数据,甚至还需要连接数据库去获取 Orders 记录,在这种场景下,没必要给 customer 集合中的所有人都带上完整的 orders,这个初始化开销是巨大的,优化点就是不加载 Orders,直到某些 customer 真的需要 Orders 时才按需灌入。
你可以自己写一段逻辑来实现 延迟初始化 ,在 .Net Framework 4.0 之后就没必要了, 因为在 System 命名空间下已经提供了 Lazy<T> ,而且还是 线程安全 的,可以使用这个类来延迟 资源密集型 的对象按需创建。
当使用 Lazy<T> 的时候,这里的 T 就是你要延迟的集合,那如何做到按需加载呢?调用 Lazy<T>.Value 即可,下面的代码片段展示了如何使用 Lazy<T> 。
Lazy<IEnumerable<Order>> orders = new Lazy<IEnumerable<Order>>(); IEnumerable<Order> result = lazyOrders.Value;
现在,考虑下面的两个类: Author 和 Blog ,一个作者可以写很多文章,所以这两个类之间是 一对多 的关系,下面的代码片段展示了这种关系。
public class Author { public int Id { get; set; } public string FirstName { get; set; } public string LastName { get; set; } public string Address { get; set; } public List<Blog> Blogs { get; set; } } public class Blog { public int Id { get; set; } public string Title { get; set; } public DateTime PublicationDate { get; set; } }
值得注意的是,关系型数据库中的 一对多 关系映射到对象模型就是 Author 类中增加一个 List Blogs 属性,使用这个属性,Author 就可以维持一个或者多个 Blog 实例对象,对吧。
现在假定在 用户界面 上仅需展示 Author 的基础信息,比如说:(firstname,lastname,address),在这种场景下,给 Author 对象加载 Blogs 集合是毫无意义的,当真的需要加载 Blogs 时,执行 Blogs.Value 即可立即执行,下面展示了 Lazy<Blog> Blogs 的用法。
public class Author { public int Id { get; set; } public string FirstName { get; set; } public string LastName { get; set; } public string Address { get; set; } public Lazy<IList<Blog>> Blogs => new Lazy<IList<Blog>>(() => GetBlogDetailsForAuthor(this.Id)); private IList<Blog> GetBlogDetailsForAuthor(int Id) { //Write code here to retrieve all blog details for an author. } }
接下来让我们看看如何使用泛型的 Lazy 实现单例模式,下面的 StateManager 是线程安全的,同时为了演示 延迟初始化 ,我使用了 静态构造函数 来确保 C# 编译器不会将它标记为 beforefieldinit 。
public sealed class StateManager { private StateManager() { } public static StateManager Instance { get { return Nested.obj; } } private class Nested { static Nested() { } internal static readonly StateManager obj = new StateManager(); } }
下面我用 Lazy<T> 来包装 StateManager,你会发现使用 Lazy<T> 来做延迟初始化真的是太简单了。。。
public class StateManager { private static readonly Lazy<StateManager> obj = new Lazy<StateManager>(() => new StateManager()); private StateManager() { } public static StateManager Instance { get { return obj.Value; } } }
可以瞄一下上面代码的 Instance 属性,它被做成只读属性了,同时也要注意 obj.Value 也是一个只读属性。
public class Lazy<T> { public T Value { get { if (_state != null) { return CreateValue(); } return _value; } } }
延迟初始化 是一个很不错的性能优化技术,它允许你将那些 资源密集型 的对象延迟到你真正需要加载的时候再加载,大家结合自己的场景尽情的使用吧!
到此这篇关于C#中Lazy如何使用的文章就介绍到这了,更多相关C#中Lazy使用内容请搜索小牛知识库以前的文章或继续浏览下面的相关文章希望大家以后多多支持小牛知识库!
译文链接:https://www.infoworld.com/article/3227207/how-to-perform-lazy-initialization-in-c.html
本文向大家介绍详解C++中shared_ptr的使用教程,包括了详解C++中shared_ptr的使用教程的使用技巧和注意事项,需要的朋友参考一下 shared_ptr是一种智能指针(smart pointer)。shared_ptr的作用有如同指针,但会记录有多少个shared_ptrs共同指向一个对象。 这便是所谓的引用计数(reference counting)。一旦最后一个这样的指针被销毁
本文向大家介绍C++中BitBlt的使用方法详解,包括了C++中BitBlt的使用方法详解的使用技巧和注意事项,需要的朋友参考一下 BitBlt 该函数对指定的源设备环境区域中的像素进行位块(bit_block)转换,以传送到目标设备环境。 原型: 参数: hdcDest:指向目标设备环境的句柄。 nXDest、nYDest:指定目标矩形区域左上角的X轴和Y轴逻辑坐标 nWidth、nH
本文向大家介绍C#中HttpWebRequest、WebClient、HttpClient的使用详解,包括了C#中HttpWebRequest、WebClient、HttpClient的使用详解的使用技巧和注意事项,需要的朋友参考一下 HttpWebRequest: 命名空间: System.Net,这是.NET创建者最初开发用于使用HTTP请求的标准类。使用HttpWebRequest可以让开发
本文向大家介绍C/C++中*和&的用法详解,包括了C/C++中*和&的用法详解的使用技巧和注意事项,需要的朋友参考一下 C++中&和*的用法一直是非常让人头疼的难点,课本博客上讲这些知识点一般都是分开讲其用法的,没有详细的总结,导致我在这方面的知识结构格外混乱,在网上找到了一篇英文文章简单总结了这两个符号的一些用法,都是一些比较基础的知识,我比较关心的函数指针,指针函数等都没有涉及到,今后有时间把
问题内容: 如何使用node.js中的C ++库? 问题答案: 看一下node-ffi。 node-ffi是一个Node.js插件,用于使用纯JavaScript加载和调用动态库。它可用于创建与本机库的绑定,而无需编写任何C ++代码。
本文向大家介绍c++ 中__declspec 的用法详解,包括了c++ 中__declspec 的用法详解的使用技巧和注意事项,需要的朋友参考一下 c++ 中__declspec 的用法如下,想要了解的继续往下看吧。 语法说明: __declspec ( extended-decl-modifier-seq ) 扩展修饰符: 1:align(#) 用__declspec(align(#))精确控制