前言
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吧~