Kotlin 使用DSL构建语法结构 看这一篇就够了~

2020-09-15 11:07:18 浏览数 (1)

前言

DSL并不是单独为Kotlin语言提供的,可能你并知道DSL是什么,但是我敢说,只要你是Android开发者,你就一定使用过并且一直在使用DSL,那么到底什么是DSL?使用DSL又可以实现怎么样的功能呢?

DSL是什么

DSL的全程是Domain Specific Language 即 领域特定语言,我们可以通过DSL语言 构建出属于我们自己的语法结构,而在Kotlin中并不只有一种方式实现DSL,而主要的实现方式就是高阶函数(如果你不了解高阶函数,也不用怕,后面我会单独一篇文章来介绍高阶函数)~

我是Android开发者,我怎么没用过DSL,我阿黄哥不信!

想想看,平时我们引入一个开源包需要怎么做呢,我们会在build.gradle中看到这样的代码:

代码语言:javascript复制
dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlin_version"
    implementation 'androidx.appcompat:appcompat:1.2.0'
    implementation 'androidx.core:core-ktx:1.3.1'
    implementation 'androidx.constraintlayout:constraintlayout:2.0.1'
    testImplementation 'junit:junit:4.12'
    androidTestImplementation 'androidx.test.ext:junit:1.1.2'
    androidTestImplementation 'androidx.test.espresso:espresso-core:3.3.0'
}

Gradle 我们都知道 它是一种基于Groovy的构建工具,上面的代码写法其实是Groovy为我们提供的DSL功能。

DSL的基础用法

接下来,我们来看,Kotlin中如何使用DSL构建自己的语法呢,要想装13 我们当然要来实现和上面一样的语法结构,那我们就来造一个吧~

首先我们新建一个类Dependency,名字是可以随便起的,只不过我们为了装13,就取的和我们经常使用的一样,声明一个list数组,为list提供添加的数据的方法,类代码如下所示:

代码语言:javascript复制
class Dependency {
    var libs = mutableListOf<String>()
    fun implementation(lib: String) {
        libs.add(lib)
    }
}

接着,我们定义一个高阶函数,参数是Dependency的扩展函数

代码语言:javascript复制
fun dependencies(block: Dependency.() -> Unit): List<String> {
    val dependency = Dependency()
    dependency.block()
    return dependency.libs
}

上面的代码,只要你了解高阶函数,肯定可以看得懂,高阶函数中的参数是Dependency的扩展函数,所以我们要先初始化一个Dependency,通过实例调用参数,就可以执行传入的Lambda表达式了,我们新建一个Test.kt,在main方法中使用如下:

代码语言:javascript复制
dependencies {
        implementation("com.huanglinqing.ll")
        implementation("com.huanglinqing.hh")
    }

怎么样,和我们在build.gradle 使用的方法很像吧

因为我们定义的方法,返回的是List,所以我们可以将结果打印出来,代码如下所示:

代码语言:javascript复制
var list = dependencies {
        implementation("com.huanglinqing.ll")
        implementation("com.huanglinqing.hh")
    }
    for (text in list) {
       println("$text")
    }

再次运行程序,结果如下所示:

代码语言:javascript复制
com.huanglinqing.ll
com.huanglinqing.hh

Process finished with exit code 0

DSL 还可以怎么用

DSL 可以将符合标准API规范的代码转化为符合人类理解的自然语言

我们以创建一个用户对象为例,新建User.kt,为了方便打印 我们重写toString方法,代码如下所示:

代码语言:javascript复制
data class User(var name: String = "", var age: Int = 0) {
    override fun toString(): String {
        return "My name is $name ,i am $age years old"
    }
}

我们仍然在Test.kt中写测试代码,来看下按照API规范我们如何来创建一个User对象

代码语言:javascript复制
val user = User("Huanglinqing", 25)
    println(user)

运行结果如下所示:

代码语言:javascript复制
My name is Huanglinqing ,i am 25 years old

Process finished with exit code 0

那么,我们如何使用DSL的方式去创建一个User对象呢,首先我们需要提供一个高阶函数

代码语言:javascript复制
fun create(block: User.() -> Unit): User {
    var user = User()
    block(user)
    return user
}

我们定义了一个类型为User扩展函数的高阶函数,通过block调用表达式的部分

所以我们可以直接这样来创建一个User对象:

代码语言:javascript复制
val user1 = create {
    name = "黄林晴"
    age = 25
}

println(user1)

我们称这种方式是更符合理解的方式,运行结果与上面一致,这里就不再演示了。

Anko插件

最后我们来简单的介绍下DSL在Kotlin中的一个框架Anko,Anko用Kotlin DSL 写的Android插件,Anko主要的作用是替代以前用XML的方式来生成UI布局。不过Jetpack推出了compose,虽然还没有正式版本,但是用的也比较多了,很多人都不了解说为什么要推荐在Activity中写布局代码 就像Flutter一样?大家都知道,Android界面是通过XML来进行布局的,一个应用中通常有多个布局,当程序运行时,XML被转化为Java代码,这里要划重点,即使不是在Java中写的代码,最终还是会转化为Java代码,这就会导致程序很耗费资源。由于Anko是直接通过Java代码来编写布局文件的,不用进行转化,因此使用Anko编写Android界面的布局会更加简单、快捷。所以我猜测,这也许为什么Jetpack要推出compose的原因之一吧。

关于Anko插件如何使用,就不讲解了,感兴趣的可直接到Github上了解:https://github.com/Kotlin/anko

写在最后

DSL的使用场景远远不止这些,其实前提就是使用好高阶函数,很多例子都讲到了使用DSL来生成HTML的代码,不过在业务中没get到他的作用,想了解的朋友可以私下和我沟通。其实不管任何一种技术,一个框架,我们不能评判他的好坏,存在即合理,推动项目开展才是王道。好了 ,DSL的基础了解就到这里了,快去愉快的装13吧~

0 人点赞