Koltin Contract DSL分析

前言

对于Kotlin DSL不熟悉的同学建议先阅读《Kotlin in Action》第11章 DSL构建

本文主要探讨Kotlin Contract DSL,熟悉源码的同学应该都不陌生.在我们经常使用的语法糖let,apply,isNullOrBlank 等中都有它们的身影出现,但是平时的开发我们可能都不会关心这个到底是做什么用的.但是随着Kotlin 1.3版本发布,contract渐渐放开了部分权限让开发者可以去使用,但是到底该如何使用?它有什么作用呢?本文将结合源码分析contract的设计、使用及其原理

Contract DSL 用武之地

我们平时开发中都会注意到编译器会对kotlin进行smartcasts,比如以下情况:

fun test(a: String?) {
    if (a != null) {
        a.length //在这里类型从String?智能转换成了String
    }
}

但是,如果你在使用自定义的函数去处理检查,那么smartcasts就会消失, 比如以下情况:

fun String?.isNotNull(): Boolean = this != null

fun test(s: String?) {
    if (s.isNotNull()) {
        s.length   //在这里不加?则无法通过编译 smartcasts消失了
    }
}

为了改善这种情况,Kotlin 1.3引入了contract的实验机制.
contract允许函数以编译器理解的方式显式地描述其行为,其实就是告诉编译器我代码是1000%是正确的,你不要再来多此一举再做检查了!.

目前,该特性广泛应用以下两种场景:

  1. 通过声明函数的调用结果与传递的参数值之间的关系来改进智能广播分析
/**
* 需要在编译选项中开启实验功能-Xuse-experimental=kotlin.Experimental
* 并且kotlin版本要>=1.3
*/
@UseExperimental(ExperimentalContracts::class)
fun require(condition: Boolean) {
 // 这是一种告诉编译器的语法方式
 // 如果函数正常返回,则condition为true
 contract { returns() implies condition }
 if (!condition) throw IllegalArgumentException("NPE")
}

fun foo(s: String?) {
 require(s is String)
 s.length  //如果s为非null值时,智能转换将会生效,后续操作将不需要再进行判空,否则将会抛出一个参数异常
}
  1. 在存在高阶函数的情况下改进变量初始化分析
/**
 * 需要在编译选项中开启实验功能-Xuse-experimental=kotlin.Experimental
 * 并且kotlin版本要>=1.3
 */
@UseExperimental(ExperimentalContracts::class)
fun synchronizeFunc(lock: Any?, block: () -> Unit) {
  //告诉编译器这个block只会执行一次
  contract { 
    callsInPlace(block, InvocationKind.EXACTLY_ONCE) 
  }
}

fun foo() {
  val x: Int
  synchronizeFunc(lock) {
    x = 42 //编译器知道这个方法会只会执行一次,不会存在对一个val变量进行多次赋值的情况
  }
  println(x) // 编译器知道lambda将被明确调用,执行初始化
  // 因此'x'被认为是在这里初始化的
}

stdlib已经大量使用了contract,所以大家不要担心之后contract是不是不会转正的问题.
contract目前表现十分稳定,大家可以放心大胆的去使用它.以下场景中,就是stdlib中的contract效果:

fun test(s: String?) {
    if (!s.isNullOrEmpty()) {
        println(s.length)  //smartcast to not-null 
    }
}
Contract DSL 设计之路

在前面的例子中,我们看到contract callsInPlace InvocationKind.EXACTLY_ONCE returns implies等一大堆平时都没有见过的方法和字段. 这些字段其实分别在EffectContractBuilder两个类中,PY关系如下:

contract.jpg

Effect: 该基础接口表示函数的调用效果,要么是直接可以观察的(正常返回的函数),要么可能是间接作用的(函数的Lambda参数调用).

ConditionalEffect: 该接口在Effect的基础之上,在观察函数的调用效果之后,某些条件是否会为true. 常用在contract{ }中指定,通过使用函数[SimpleEffect.implies]将布尔表达式附加到另一个[SimpleEffect]效果,

SimpleEffect: 表示函数调用之后可以观察的效果

  • implies(booleanExpression: Boolean): ConditionalEffect: 指定此效果在观察时保证[booleanExpression]为true。 注意:[booleanExpression]只能接受布尔表达式的子集,

例如以下示例:

public inline fun CharSequence?.isNullOrEmpty(): Boolean {
    contract {
        returns(false) implies (this@isNullOrEmpty != null)
    }

    return this == null || this.length == 0
}

implies接收了(this@isNullOrEmpty != null)表示当前字符串为null时,直接告诉编译器当前条件为false,这有什么用呢?如果我们使用了!xxx.isNullOrEmpty,那么编译就会智能认定后续的xxx肯定不为null,则不会在提示你去加?进行防御编程.

关于为什么是returns而不是return?我们在后面讲解ContractBuilder时详细介绍.

Returns: 该接口描述函数正常返回给定值的情况
ReturnsNotNull: 该接口描述函数正常返回任何非null返回值的情况
CallsInPlace: 该接口表示调用函数式参数(lambda表达式参数)的效果,并且函数式参数(lambda表达式参数)只能在自己函数被调用期间被调用,当自己函数被调用结束后,函数式参数(lambda表达式参数)不能被执行.

大家看是不是觉得很眼熟啊?

方法 返回值
returns Returns
returnsNotNull ReturnsNotNull
callsInPlace CallsInPlace

这么一看,其实就是对应方法调用之后的返回效果.
我们再来看一看这些方法都是什么意思,怎么用的?

ContractBuilder

注解ExperimentalContracts: 由于目前合同还是属于实现API,所以在声明合同的地方加上该注解.


returns(): 描述函数正常返回(无返回值)但没有抛出任何异常的情况。不能单独使用,需要使用[SimpleEffect.Implies]函数来描述在这种情况下发生的条件效果。

fun embedVariable(x: Any, b: Boolean) {
    contract {
        //当b==true并且x是一个非空String时正常返回(无返回值)
        returns() implies (b && x is String)
    }
}

returns(value: Any?): 描述函数以指定的return [value]正常返回的情况。同样的也要配合使用[SimpleEffect.Implies]函数来描述在这种情况下发生的条件效果
[value]的可能值限于truefalsenull

fun threeReturnsValue(b: Boolean?) {
    contract {
        returns(true) implies (b == true)
        returns(null) implies (b == null)
        returns(false) implies (b == false)
    }
}

returnsNotNull(): 描述函数正常返回任何非“null”值的情况。使用[SimpleEffect.Implies]函数来描述在这种情况下发生的条件效果。

fun threeReturnsValue(b: Boolean) {
    contract {
        returnsNotNull() implies (b != null)
        returns(true) implies (b)
        returns(false) implies (!b)
    }
}

callsInPlace: 用于在适当的位置调用函数参数[lambda],该合同有以下规定:

  • 函数[lambda]只能在所有者函数调用期间调用,并且在完成所有者函数调用后不会调用它;
  • (可选)函数[lambda]被调用[kind]参数指定的次数,请参阅[InvocationKind]枚举以获取可能的值。

声明callsInPlace效果的函数必须是inline。

@kotlin.internal.InlineOnly
public inline fun <T> T.apply(block: T.() -> Unit): T {
    contract {
        //该block在所有者函数调用期间只会执行一次
        callsInPlace(block, InvocationKind.EXACTLY_ONCE)
    }
    block()
    return this
}

InvocationKind
  • AT_MOST_ONCE: 函数参数将被调用一次或根本不被调用
  • AT_LEAST_ONCE: 函数参数将被调用一次或多次。
  • EXACTLY_ONCE: 函数参数将被调用一次
  • UNKNOWN: 函数参数就地调用,但不知道可以调用多少次。
总结

本文的内容收益主要是帮助我们去理解底层的实现与原理,仔细研究就会发现套路其实就那么几种,但是用的好就是奇淫技巧. 为了我们以后使用的得心应手,一起来研究吧 ~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,377评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,390评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,967评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,344评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,441评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,492评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,497评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,274评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,732评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,008评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,184评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,837评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,520评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,156评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,407评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,056评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,074评论 2 352

推荐阅读更多精彩内容