راهنمای ایجاد DSL در کاتلین — به زبان ساده

آخرین به‌روزرسانی: ۲۳ فروردین ۱۳۹۹
زمان مطالعه: ۲ دقیقه
DSL در کاتلین

یکی از قابلیت‌های واقعاً مفید کاتلین امکان ساخت DSL سفارشی است. در این مقاله به بررسی روش ایجاد DSL در کاتلین می‌پردازیم. به این منظور تنها به چهار چیز نیاز دارید:

  • نمادگذاری میانوندی
  • متدهای اکستنشن
  • لامبدا
  • لامبدا با گیرنده

در ادامه این موارد را یک به یک بررسی می‌کنیم.

نمادگذاری میانوندی

در «نمادگذاری میانوندی» (Infix Notation) براکت‌ها و نقطه‌ها را حذف می‌کنیم:

class Car {
   fun drive(miles: Int) {
      ...
   }
}
val car = Car()
car.drive(10) // normal stuff

کافی است نمادگذاری میانوندی را پیش از اعلان متد اضافه کنیم تا کار کند:

class Car {
   infix fun drive(miles: Int) {
      ...
   }
}
val car = Car()
car drive 10 // no brackets! no dots!

متدهای اکستنشن

با استفاده از متدهای اکستنشن می‌توان تابع‌های جدید را به هر کلاس بدون نیاز به ارث‌بری اضافه کرد.

val name = "Ahmed Riz"
name.shout() // won't compile as shout() is not part of String class

برای این که آن را بدون ارث‌بری به عنوان بخشی از کلاس String دربیاوریم، باید متد shout را بدون پیشوند String ایجاد کنیم.

fun String.shout() {
   println("$this !") // this refers to the value of string itself
}
val name = "Ahmed Riz"
name.shout()  // works! and prints out: Ahmed Riz !

متدهای اکستنشن دامنه پروژه‌ای دارند. می‌توان از هر جایی درون پروژه به آن‌ها دسترسی داشت.

لامبدا

با استفاده از لامبداها می‌توان «الفاظ تابعی» (function literals) را به تابع‌های مرتبه بالاتر ارسال کرد.

// a higher order function
fun process(value: Int, operation: (Int) -> Unit) {
    operation(value)
}
// when calling, we can pass in a lambda expression
process(10, { value -> println(value) })
// and because it's the last param, we can extract it out
// just to clean things up even further 
process(10) { value -> println(value) }

لامبدا با گیرنده

با استفاده از لامبدا به همراه گیرنده می‌توان الفاظ تابعی را به تابع‌های مرتبه بالاتر اما با نوع گیرنده ارسال کرد.

// slight change to the previous higher order function
// now instead of operation: (Int) -> Unit
// we'll do operation: Int.() -> Unit
// This makes Int the receiver type
fun process(value: Int, operation: Int.() -> Unit) {
    value.operation()
}
// usage now becomes
process(10) { println(this) } // prints out: 10

ترکیب همه موارد

در این بخش با روش ساخت یک DSL خاص که صرفاً به منظور نمایش شیوه کار ایجاد می‌شود آشنا می‌شویم.

val myProfile = "Ahmed Rizwan" profile {
    age = 90
    phone = "123 456 789"      
}

کد تبدیل آن به یک DSL معتبر به سادگی زیر است:

class Profile(
    val name: String,
    var age: Int? = null,
    var phone: String? = null
)

infix fun String.profile(create: Profile.() -> Unit): Profile {
    val profile = Profile(this)
    profile.create()
    return profile
}

سخن پایانی

بدین ترتیب موفق شدیم یک DSL سفارشی در کاتلین ایجاد کنیم. با این که این DSL ممکن است در دنیای واقعی چندان مفید نباشد، اما امیدواریم به این ترتیب ایده خوبی از شیوه استفاده از این قابلیت‌های متفاوت کاتلین برای ساخت DSL سفارشی برای خودتان به دست آورده باشید.

اگر این مطلب برای شما مفید بوده است‌، آموزش‌های زیر نیز به شما پیشنهاد می‌شوند:

==

بر اساس رای ۴ نفر
آیا این مطلب برای شما مفید بود؟
شما قبلا رای داده‌اید!
اگر بازخوردی درباره این مطلب دارید یا پرسشی دارید که بدون پاسخ مانده است، آن را از طریق بخش نظرات مطرح کنید.
منابع:
android.jlelse

نظر شما چیست؟

نشانی ایمیل شما منتشر نخواهد شد. بخش‌های موردنیاز علامت‌گذاری شده‌اند *