Android设计模式(二) 建造者模式

建造者模式解决的就是将一个复杂对象的构建与它的表示分离,从而使对象的构建过程能有不同的表示。什么意思?

fun main(args: Array<String>) {
    MacBook(15,8,16,512,true,"test","")
}

class MacBook(private var screenSize:Int,
              private var cpuCore:Int,
              private var ramCapacity:Int,
              private var diskCapacity:Int,
              private var appleCare:Boolean,
              private var engraveWords:String,
              private var protectingShell:String)

面对这种过长的参数构造函数,我们需要通过建造者模式来改造。

不管是之前提到的工厂模式还是单例,都没解决扩展大量可选参数的问题,例如上述我们在购买MacBook的时候都会进行可选产品定制,例如屏幕尺寸,处理器种类 ,内存大小,硬盘大小 ,Applecare,铭刻内容,配件...等等等

在业务中我们也经常使用重叠构造器(telescoping constructor),先提供一个只有必要参数的构造函数,在提供其他不同参数组合的构造,但是如果参数过长是不利于维护的。所以我们用建造者模式改造:

fun main(args: Array<String>) {
    MacBook.Builder()
            .setScreenSize(15)
            .setCpuCore(8)
            .setRamCapacity(16)
            .setDiskCapacity(512)
            .setAppleCare(true)
            .build()
}

class MacBook private constructor(var screenSize: Int,
                                  var cpuCore: Int,
                                  var ramCapacity: Int,
                                  var diskCapacity: Int,
                                  var appleCare: Boolean) {

    class Builder(private var screenSize: Int? = null,
                  private var cpuCore: Int? = null,
                  private var ramCapacity: Int? = null,
                  private var diskCapacity: Int? = null,
                  private var appleCare: Boolean = false) {

        fun setScreenSize(screenSize: Int?): Builder {
            this.screenSize = screenSize
            return this
        }

        fun setCpuCore(cpuCore: Int?): Builder {
            this.cpuCore = cpuCore
            return this
        }

        fun setRamCapacity(ramCapacity: Int?): Builder {
            this.ramCapacity = ramCapacity
            return this
        }

        fun setDiskCapacity(diskCapacity: Int?): Builder {
            this.diskCapacity = diskCapacity
            return this
        }

        fun setAppleCare(appleCare: Boolean): Builder {
            this.appleCare = appleCare
            return this
        }

        fun build(): MacBook {
            return MacBook(screenSize!!, cpuCore!!, ramCapacity!!, diskCapacity!!, appleCare)
        }

    }
}

步骤:

  • 定义嵌套类Builder负责创建MacBook对象
  • 私有化类MacBook的构造函数,确保无法直接通过MacBook声明实例
  • 通过在Builder类中定义的可选属性来定制对象创建的内容
  • 调用build()方法返回一个MacBook实例

这样比直接在构造函数中配置参数优雅了不少,创建实例时,只需要在对set方法进行赋值即可。

不足:

  • 业务需求参数非常多时,代码依旧会显得比较长
  • build()方法经常会忘记调用
  • 创建实例时必须先创建其构造器。

这是Java中典型的建造者模式,但在kotlin中我们其实可以尽量避免使用它,因为建造者模式本质上也是模拟了具名的可选参数 ,例如Flutter中的Dart使用 { }来包裹具名参数传递的时候,和参数顺序无关,当然kotlin也支持这种特性

void printTest(String name1, {String name2, String name3}) { }

具名可选参数

kotlin函数与构造器都支持具名可选参数,我们直接利用data class声明一个数据类 :

 fun main(args: Array<String>) {
    MacBook(screenSize = 15)
    MacBook(appleCare = true ,screenSize = 15,cpuCore = 8)
}
data class MacBook(val screenSize: Int? = null,
                   val cpuCore: Int? = null,
                   val ramCapacity: Int? = null,
                   val diskCapacity: Int? = null,
                   val appleCare: Boolean? = null)

这样我们声明对象时,每个参数名都是显式的,不用按照顺序定义参数内容,并且参数都由val声明更加安全

利用requair进行参数行为约束

为了让业务更加安全,我们经常会利用建造者模式中的属性进行业务行为约束

fun main(args: Array<String>) {
    MacBook(cpuCore = 38)
}

class MacBook(val screenSize: Int? = null,
                   val cpuCore: Int? = null,
                   val ramCapacity: Int? = null,
                   val diskCapacity: Int? = null,
                   val appleCare: Boolean? = null) {
    init {
        require(cpuCore!! <= 32) {
            "CPU核心数超过定制需求"
        }
    }
}

这样当我们传入的参数定义符合requair中的约束,就会抛出异常:

Exception in thread "main" java.lang.IllegalArgumentException: CPU核心数超过定制需求

当然这为了说明requair示例,具体这种业务还是需要数据配置例如枚举,自定义注解等进行详细约束的

数据模型独立

这里多说一句,上述requair没有使用data class,Kotlin中带有数据类的模型已经比Java同类的模型更精简和简洁。用一个data关键字抽象了所有参数的getters, setters、toString()和copy()方法,使我们的数据类能够反映出他们唯一需要关注的事情—数据模型独立。

所以我们在业务开发时可以借助Kotlin的扩展将业务或转换逻辑与数据模型分离,例如在我们在企业级IM应用中需要对用户页面或者访问内容进行动态分发

data class RongCloudCast(
        val uuid: String,
        val departments: String,
        val category: String,
        val website: String,
        val version: String,
        val hashCode: String
) {
    // 路由跳转基础路径
    fun getRouterBaseUrl(): String = "$departments/$category"
    // 允许传递给WebView进程的URL
    fun getWebViewEntryUrl(): String =  "https://$website/$version/$hashCode"
}

相较于这种还是由扩展功能完成拼接逻辑

data class RongCloudCast(
        val uuid: String,
        val departments: String,
        val category: String,
        val website: String,
        val version: String,
        val hashCode: String
)

fun RongCloudCast.getRouterBaseUrl(): String = "$departments/$category"
fun RongCloudCast.getWebViewEntryUrl(): String = "https://$website/$version/$hashCode"
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352

推荐阅读更多精彩内容