Kotlin 基础(二)- DSL

所谓 DSL 领域专用语言(Domain Specified Language/ DSL),其基本思想是“求专不求全”,不像通用目的语言那样目标范围涵盖一切软件问题,而是专门针对某一特定问题的计算机语言。

Kotlin DSL 定义:使用 Kotlin 语言开发的,解决特定领域问题,具备独特代码结构的 API 。

一、DSL

DSL(domain specific language),即领域专用语言:专门解决某一特定问题的计算机语言,比如大家耳熟能详的 SQL 和正则表达式。使用DSL的编程风格,可以让程序更加简单干净、直观简洁。当然,我们也可以创建自己的 DSL。相对于传统的 API, DSL 更加富有表现力、更符合人类语言习惯。

如果为解决某一特定领域问题就创建一套独立的语言,开发成本和学习成本都很高,因此便有了内部 DSL 的概念。所谓内部 DSL,便是使用通用编程语言来构建 DSL,比如,Kotlin DSL。

1.1 常见的 DSL

常见的 DSL 在很多领域都能看到,例如:

  • 软件构建领域 Ant
  • UI 设计师 HTML
  • 硬件设计师 VHDL

1.2 通用编程语言 vs DSL

通用编程语言(如 Java、Kotlin、Android等),往往提供了全面的库来帮助开发者开发完整的应用程序,而 DSL 只专注于某个领域,比如 SQL 仅支持数据库的相关处理,而正则表达式只用来检索和替换文本,我们无法用 SQL 或者正则表达式来开发一个完整的应用。

  • DSL 供非程序员使用,供领域专家使用;
  • DSL 有更高级的抽象,不涉及类似数据结构的细节;
  • DSL 表现力有限,其只能描述该领域的模型,而通用编程语言能够描述任意的模型;

无论是通用编程语言,还是领域专用语言,最终都是要通过 API 的形式向开发者呈现。良好的、优雅的、整洁的、一致的 API 风格是每个优秀开发者的追求,而 DSL 往往具备独特的代码结构和一致的代码风格。

二、Kotlin DSL 在 Android 开发的应用

2.1 Anko

Anko 是一个 DSL (Domain-Specific Language), 它是 JetBrains 出品的,用 Kotlin 开发的安卓框架。它主要的目的是用来替代以前 XML 的方式来使用代码生成 UI 布局。

2.1.1 使用

Anko 中, 不需要继承其他奇怪的类,只要标准的 Activity, Fragment,FragmentActivity 或者其他任意的类

首先, 在使用 Anko 的 DSL 的类中导入 org.jetbrains.anko.* .

DSL 可以在 onCreate()中使用:

override fun onCreate(savedInstanceState: Bundle?) {
    super<Activity>.onCreate(savedInstanceState)

    verticalLayout {
        padding = dip(30)
        editText {
            hint = "Name"
            textSize = 24f
        }
        editText {
            hint = "Password"
            textSize = 24f
        }
        button("Login") {
            textSize = 26f
        }
    }
}

不需要显示的调用 setContentView(R.layout.something), Anko 自动为 Activity(只会对Activity)进行 set content view

padding, hint 和 textSize 是 扩展属性. 大多数 View 具有这些属性,允许使用 text = “Some text” 代替 setText(“Some text”).

verticalLayout (一个竖直方向的 LinearLayout), editText 和 button

扩展函数. 这些函数存在与ANdroid 框架中的大部View中, Activities, Fragments ( android.support 包中的) 甚至 Context同样适用.

如果有一个 Context 实例, 可以写出下面的DSL结构:

val name = with(myContext) {
    editText {
        hint = "Name"
    }
}

变量 name 成为了 EditText类型.

2.1.2 Helper 方法

你可能注意到了,前面 button 方法接了一个字符串参数,这样的Helper方法同样使用与 TextView, EditText, Button , ImageView.

如果你不需要 View 其他的属性,你可以省略 {} 直接写 button(“Ok”) 或只有 button():

verticalLayout {
    button("Ok")
    button("Cancel")
}
2.1.3 Layouts 和 LayoutParams

在父布局中布局控件可能需要使用 LayoutParams.xml 中长这样:

<ImageView 
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android_layout_marginLeft="5dip"
    android_layout_marginTop="10dip"
    android:src="@drawable/something" />

Anko 中, 在 View 的后面使用 lparams 来实现类似与 xml 的 LayoutParams。

linearLayout {
    button("Login") {
        textSize = 26f
    }.lparams(width = wrapContent) {
        horizontalMargin = dip(5)
        topMargin = dip(10)
    }
}

如果指定了 lparams,但是没有指定 width 或者 height, 默认是 WRAP_CONTENT,但是你可以自己通过使用 named arguments指定.

注意下面一些方便的属性:

  • horizontalMargin 同时设置 left 和 right margins
  • verticalMargin 同时设置 top 和 bottom
  • margin 同时设置4个方向的 margins

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_height="match_parent" android:layout_width="match_parent"> <EditText android:id="@+id/todo_title" android:layout_width="match_parent" android:layout_heigh="wrap_content" android:hint="@string/title_hint" /> <!-- Cannot directly add an inline click listener as onClick delegates implementation to the activity --> <Button android:layout_width="match_parent" android:layout_height="wrap_content" android:text="@string/add_todo" /> </LinearLayout>

使用 Anko 之后,可以用代码实现布局,并且 button 还绑定了点击事件。


verticalLayout { var title = editText { id = R.id.todo_title hintResource = R.string.title_hint } button { textResource = R.string.add_todo onClick { view -> { // do something here title.text = "Foo" } } } }

可以看到 DSL 的一个主要优点在于,它需要很少的时间即可理解和传达某个领域的详细信息。

override fun onCreate(savedInstanceState: Bundle?) { 
   super.onCreate(savedInstanceState) 
   verticalLayout { 
       padding = dip(30) 
       editText { 
         hint = "Name" 
         textSize = 24f 
       } 
       editText { 
         hint = "Password" 
         textSize = 24f 
       } 
       button("Login") { 
         textSize = 26f 
       } 
   } 
 }

2.1.4 弹窗
 alert("Hi, I'm Roy", "Have you tried turning it off and on again?") { 
     yesButton { toast("Oh…") } 
     noButton {} 
 }.show()

2.1.5 异步

 doAsync { 
     // Long background task 
     uiThread { 
         result.text = "Done" 
     } 
 } 

Refer