前言
DSL并不是单独为Kotlin语言提供的,可能你并知道DSL是什么,但是我敢说,只要你是Android开发者,你就一定使用过并且一直在使用DSL,那么到底什么是DSL?使用DSL又可以实现怎么样的功能呢?
DSL是什么
DSL的全程是Domain Specific Language 即 领域特定语言,我们可以通过DSL语言 构建出属于我们自己的语法结构,而在Kotlin中并不只有一种方式实现DSL,而主要的实现方式就是高阶函数(如果你不了解高阶函数,也不用怕,后面我会单独一篇文章来介绍高阶函数)~
我是Android开发者,我怎么没用过DSL,我阿黄哥不信!
想想看,平时我们引入一个开源包需要怎么做呢,我们会在build.gradle中看到这样的代码:
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提供添加的数据的方法,类代码如下所示:
class Dependency { var libs = mutableListOf<String>() fun implementation(lib: String) { libs.add(lib) } }
接着,我们定义一个高阶函数,参数是Dependency的扩展函数
fun dependencies(block: Dependency.() -> Unit): List<String> { val dependency = Dependency() dependency.block() return dependency.libs }
上面的代码,只要你了解高阶函数,肯定可以看得懂,高阶函数中的参数是Dependency的扩展函数,所以我们要先初始化一个Dependency,通过实例调用参数,就可以执行传入的Lambda表达式了,我们新建一个Test.kt,在main方法中使用如下:
dependencies { implementation("com.huanglinqing.ll") implementation("com.huanglinqing.hh") }
怎么样,和我们在build.gradle 使用的方法很像吧
因为我们定义的方法,返回的是List,所以我们可以将结果打印出来,代码如下所示:
var list = dependencies { implementation("com.huanglinqing.ll") implementation("com.huanglinqing.hh") } for (text in list) { println("$text") }
再次运行程序,结果如下所示:
com.huanglinqing.ll com.huanglinqing.hh Process finished with exit code 0
DSL 还可以怎么用
DSL 可以将符合标准API规范的代码转化为符合人类理解的自然语言
我们以创建一个用户对象为例,新建User.kt,为了方便打印 我们重写toString方法,代码如下所示:
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对象
val user = User("Huanglinqing", 25) println(user)
运行结果如下所示:
My name is Huanglinqing ,i am 25 years old Process finished with exit code 0
那么,我们如何使用DSL的方式去创建一个User对象呢,首先我们需要提供一个高阶函数
fun create(block: User.() -> Unit): User { var user = User() block(user) return user }
我们定义了一个类型为User扩展函数的高阶函数,通过block调用表达式的部分
所以我们可以直接这样来创建一个User对象:
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吧~