by lazy 的作用

  延迟属性(lazy properties) 是 Kotlin 标准库中的标准委托之一,可以通过 by lazy 来实现。

  其中,lazy() 是一个函数,可以接受一个 Lambda 表达式作为参数,第一次调用时会执行 Lambda 表达式,以后调用该属性会返回之前的结果。

  例如下面的代码:

  val str: String by lazy{

  println(aaron)

  println(cafei)

  tony // 最后一行为返回值

  }

  fun main(args: Array) {

  println(str)

  println(-----------)

  println(str)

  }

  执行结果:

  aaron

  cafei

  tony

  -----------

  tony

  因为 lazy() 的最后一行,返回的值即为 str 的值,以后每次调用 str 都可以直接返回该值。

  源码分析

  从 lazy() 开始分析源码:

  public actual funlazy(initializer: () - T): Lazy= SynchronizedLazyImpl(initializer)

  actual 是 Kotlin 的关键字表示多平台项目中的一个平台相关实现。

  lazy 函数的参数是 initializer,它是一个函数类型。lazy 函数会创建一个 SynchronizedLazyImpl 类,并传入 initializer 参数。

  下面是 SynchronizedLazyImpl 的源码:

  private class SynchronizedLazyImpl(initializer: () - T, lock: Any? = null) : Lazy, Serializable {

  private var initializer: (() - T)? = initializer

  @Volatile private var _value: Any? = UNINITIALIZED_VALUE

  // final field is required to enable safe publication of constructed instance

  private val lock = lock ?: this

  override val value: T

  get() {

  val _v1 = _value

  if (_v1 !== UNINITIALIZED_VALUE) {

  @Suppress(UNCHECKED_CAST)

  return _v1 as T

  }

  return synchronized(lock) {

  val _v2 = _value

  if (_v2 !== UNINITIALIZED_VALUE) {

  @Suppress(UNCHECKED_CAST) (_v2 as T)

  } else {

  val typedValue = initializer!!()

  _value = typedValue

  initializer = null

  typedValue

  }

  }

  }

  override fun isInitialized(): Boolean = _value !== UNINITIALIZED_VALUE

  override fun toString(): String = if (isInitialized()) value.toString() else Lazy value not initialized yet.

  private fun writeReplace(): Any = InitializedLazyImpl(value)

  }

  可以看到 SynchronizedLazyImpl 实现了 Lazy、Serializable 接口,它的 value 属性重载了 Lazy 接口的 value。

  Lazy 接口的 value 属性用于获取当前 Lazy 实例的延迟初始化值。一旦初始化后,它不得在此 Lazy 实例的剩余生命周期内更改。

  public interface Lazy{

  /**

  * Gets the lazily initialized value of the current Lazy instance.

  * Once the value was initialized it must not change during the rest of lifetime of this Lazy instance.

  */

  public val value: T

  /**

  * Returns `true` if a value for this Lazy instance has been already initialized, and `false` otherwise.

  * Once this function has returned `true` it stays `true` for the rest of lifetime of this Lazy instance.

  */

  public fun isInitialized(): Boolean

  }

  所以 SynchronizedLazyImpl 的 value 属性只有 get() 方法,没有 set() 方法。

  value 的 get() 方法会先判断 _value 属性是否是 UNINITIALIZED_VALUE,不是的话会返回 _value 的值。

  _value 使用@Volatile注解标注,相当于在 Java 中 使用 volatile 修饰 _value 属性。volatile 具有可见性、有序性,因此一旦 _value 的值修改了,其他线程可以看到其最新的值。

  SynchronizedLazyImpl 的 _value 属性存储了 initializer 的值。

  如果 _value 的值等于 UNINITIALIZED_VALUE,则调用 initializer 来获取值,通过synchronized来保证这个过程是线程安全的。

  lazy() 方法还有一个实现,它比起上面的方法多一个参数类型 LazyThreadSafetyMode。

  public actual funlazy(mode: LazyThreadSafetyMode, initializer: () - T): Lazy=

  when (mode) {

  LazyThreadSafetyMode.SYNCHRONIZED - SynchronizedLazyImpl(initializer)

  LazyThreadSafetyMode.PUBLICATION - SafePublicationLazyImpl(initializer)

  LazyThreadSafetyMode.NONE - UnsafeLazyImpl(initializer)

  }

  SYNCHRONIZED 使用的是 SynchronizedLazyImpl 跟之前分析的 lazy() 方法是一致的,PUBLICATION 使用的是 SafePublicationLazyImpl,而 NONE 使用的是 UnsafeLazyImpl。

  其中,UnsafeLazyImpl 不是线程安全的,而其他都是线程安全的。

  SafePublicationLazyImpl 使用AtomicReferenceFieldUpdater来保证 _value 属性的原子操作。毕竟,volatile 不具备原子性。

  private class SafePublicationLazyImpl(initializer: () - T) : Lazy, Serializable {

  @Volatile private var initializer: (() - T)? = initializer

  @Volatile private var _value: Any? = UNINITIALIZED_VALUE

  // this final field is required to enable safe publication of constructed instance

  private val final: Any = UNINITIALIZED_VALUE

  override val value: T

  get() {

  val value = _value

  if (value !== UNINITIALIZED_VALUE) {

  @Suppress(UNCHECKED_CAST)

  return value as T

  }

  val initializerValue = initializer

  // if we see null in initializer here, it means that the value is already set by another thread

  if (initializerValue != null) {

  val newValue = initializerValue()

  if (valueUpdater.compareAndSet(this, UNINITIALIZED_VALUE, newValue)) {

  initializer = null

  return newValue

  }

  }

  @Suppress(UNCHECKED_CAST)

  return _value as T

  }

  override fun isInitialized(): Boolean = _value !== UNINITIALIZED_VALUE

  override fun toString(): String = if (isInitialized()) value.toString() else Lazy value not initialized yet.

  private fun writeReplace(): Any = InitializedLazyImpl(value)

  companion object {

  private val valueUpdater = java.util.concurrent.atomic.AtomicReferenceFieldUpdater.newUpdater(

  SafePublicationLazyImpl::class.java,

  Any::class.java,

  _value

  )

  }

  }

  因此 SafePublicationLazyImpl 支持同时多个线程调用,并且可以在全部或部分线程上同时进行初始化。但是,如果某个值已由另一个线程初始化,则将返回该值而不执行初始化。

  总结

  lateinit 修饰的变量也可以延迟初始化,但并不是不用初始化,它需要在生命周期流程中进行获取或者初始化。

  lateinit和by lazy的区别:

  lateinit 只能用于修饰变量 var,不能用于可空的属性和 Java 的基本类型。

  lateinit 可以在任何位置初始化并且可以初始化多次。

  lazy 只能用于修饰常量 val,并且 lazy 是线程安全的。

  lazy 在第一次被调用时就被初始化,以后调用该属性会返回之前的结果。

从源码角度分析 Kotlin by lazy 的实现的更多相关文章

  1. 源码角度分析-newFixedThreadPool线程池导致的内存飙升问题

    前言 使用无界队列的线程池会导致内存飙升吗?面试官经常会问这个问题,本文将基于源码,去分析newFixedThreadPool线程池导致的内存飙升问题,希望能加深大家的理解. (想自学习编程的小伙伴请 ...

  2. 从源码角度分析 MyBatis 工作原理

    一.MyBatis 完整示例 这里,我将以一个入门级的示例来演示 MyBatis 是如何工作的. 注:本文后面章节中的原理.源码部分也将基于这个示例来进行讲解.完整示例源码地址 1.1. 数据库准备 ...

  3. Android的Message Pool是什么——源码角度分析

    原文地址: http://blog.csdn.net/xplee0576/article/details/46875555 Android中,我们在线程之间通信传递通常采用Android的消息机制,而 ...

  4. 【原创】源码角度分析Android的消息机制系列(三)——ThreadLocal的工作原理

    ι 版权声明:本文为博主原创文章,未经博主允许不得转载. 先看Android源码(API24)中对ThreadLocal的定义: public class ThreadLocal<T> 即 ...

  5. 【原创】源码角度分析Android的消息机制系列(五)——Looper的工作原理

    ι 版权声明:本文为博主原创文章,未经博主允许不得转载. Looper在Android的消息机制中就是用来进行消息循环的.它会不停地循环,去MessageQueue中查看是否有新消息,如果有消息就立刻 ...

  6. Java面试题 从源码角度分析HashSet实现原理?

    面试官:请问HashSet有哪些特点? 应聘者:HashSet实现自set接口,set集合中元素无序且不能重复: 面试官:那么HashSet 如何保证元素不重复? 应聘者:因为HashSet底层是基于 ...

  7. Adroid学习之 从源码角度分析-禁止使用回退按钮方案

    有时候,不能让用户进行回退操作,如何处理? 查看返回键触发了哪些方法.在打开程序后把这个方法禁止了. 问题:程序在后台驻留,这样就会出现,其他时候也不能使用回退按钮.如何处理,在onpase()时方法 ...

  8. 【原创】源码角度分析Android的消息机制系列(六)——Handler的工作原理

    ι 版权声明:本文为博主原创文章,未经博主允许不得转载. 先看Handler的定义: /** * A Handler allows you to send and process {@link Mes ...

  9. 【原创】源码角度分析Android的消息机制系列(四)——MessageQueue的工作原理

    ι 版权声明:本文为博主原创文章,未经博主允许不得转载. MessageQueue,主要包含2个操作:插入和读取.读取操作会伴随着删除操作,插入和读取对应的方法分别为enqueueMessage和ne ...

随机推荐

  1. 网络爬虫值scrapy框架基础

    简介 Scrapy是一个高级的Python爬虫框架,它不仅包含了爬虫的特性,还可以方便的将爬虫数据保存到csv.json等文件中. 首先我们安装Scrapy. 其可以应用在数据挖掘,信息处理或存储历史 ...

  2. Python 能干什么

    二.Python 只适合测试? 关于Python是一种什么样的语言,这里不打算说对象.类之类的术语.我们可以先来看一看,时至今日 Python 都在哪些领域里得以应用: 电信基础设施 (Twilio) ...

  3. PAT 1096 Consecutive Factors[难]

    1096 Consecutive Factors (20 分) Among all the factors of a positive integer N, there may exist sever ...

  4. cnn for qa

    最近在做QA系统,用tensorflow做了些实验,下面的的是一个cnn的评分网络.主要参考了<APPLYING DEEP LEARNING TO ANSWER SELECTION: A STU ...

  5. SQL Server OBJECTPROPERTY使用方法

    OBJECTPROPERTY 返回有关当前数据库中的模式作用域对象的信息.此函数不能用于不是模式范围的对象,例如数据定义语言(DDL)触发器和事件通知. OBJECTPROPERTY 语法: OBJE ...

  6. Sqlserver生成带数据的脚本

    右键数据库—>任务—>生成脚本 下一步 选择要导出数据库,下一步 编写数据脚本选择True,下一步 选择要导出的表,下一步 最后点击完成即可.

  7. PKU 2002 Squares(二维点哈希+平方求余法+链地址法)

    题目大意:原题链接 给定平面上的N个点,求出这些点一共可以构成多少个正方形. 解题思路: 若正方形为ABCD,A坐标为(x1, y1),B坐标为(x2, y2),则很容易可以推出C和D的坐标.对于特定 ...

  8. [转]关于Navicat和MYSQL字符集不统一出现的中文乱码问题

    原文链接:关于Navicat和MYSQL字符集不统一出现的中文乱码问题 最近遇到一串关于MYSQL中文乱码的问题,问题背景是这样的: 在此之前,服务器上安装好MySQL之后就立马重新配置了字符集为ut ...

  9. PCLK怎么获得?

    1.PCLK是由MCLK进行分频而来...... 2.PCLK是个时钟,通过寄存器只能调节它的频率什么的,它是控制像素输出的一个时钟: 3.在曝光时间的算法中需要知道PCLK的值,是因为在sensor ...

  10. 有道云笔记配合MPic+七牛云 自制MarkDown文档图床(适用Typora)

    注:从有道云笔记v6.5开始,有道云笔记会员可以使用MarkDown有道自带的图床.(但是非会员可以采用下面的七牛云图床+MarkDown方法) 0x00 前言 一直用有道云笔记,粘贴图片,做笔记没问 ...