建造者设计模式 + 高阶函数 => DSL

这篇具有很好参考价值的文章主要介绍了建造者设计模式 + 高阶函数 => DSL。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

该设计模式适用于创建复杂对象,该复杂对象通常是由各个部分的子对象用一定的算法或者步骤构成,针对每个子对象内部算法和步骤通常是稳定的,但是该复杂对象的确实由于不同的需求而选择使用不同的子对象进行组装。对于构建该复杂的对象,通常可以使用builder设计模式。而对于kotlin语言,结合高阶函数所实现的建造者设计模式算是DSL代码分享的实践。

比如我们如果想要创建一个Server类:

class Server() {
	var port: Int
	var address: String
	...
}

但是考虑到创建该对象比较复杂(该对象的成员比较多),且每一个成员的最终取值需要一定的算法策略,为了减少构造函数的参数,我们采用为该类添加一个建造者类,通过建造者类来创建该Server对象,而不是直接new该对象,为了在使用中我们直观感受到该建造者类是专门为Server类服务,故我们将该建造者类声明为该Server的内部类。

class Server(val serverBuilder: ServerBuilder) {
    class ServerBuilder {
        private var port: Int = 8080
        private var address: String = ""
        
        fun port(init: ServerBuilder.() -> Int) = apply { port = init() }
        fun address(init: ServerBuilder.() -> String) = apply { address = init() }

        
        fun build(): Server = Server(this)
    }
}

我们仔细观察下这个建造者类:ServerBuilder,因为我们是要该类帮我们最终构建Server类,那么我们就要求Server Builder要包含Server应该包含的所有的成员(port、address等),且这些成员都要设置成可变的,可重新被赋值的,即var。
同时为这些成员都增加一个对应的方法,方便从外部注入值。我们仔细观察下port和address方法,以port方法为例(address方法结构和port方法类似)。

port方法的参数接收一个参数,该参数是一个lambda表达式(一个高阶函数),该port的方法的返回值是apply的返回值,而我们知道kotlin的apply方法一般是作用于一个对象的,且最终的返回值就是这个对象,很明显此处的apply方法省略了this,apply的返回值就是当前的ServerBuilder对象,完成的写法应该是

this.apply {

}

其等价于:

fun port(init: ServerBuilder.() -> Int): ServerBuilder {
    init()
    return this
}

apply的内部则是将lambda的表达式的返回值赋值给了该ServerBuilder对象的port成员。我们再来看下port方法所接收的lambda表达式:

ServerBuilder.() -> Int

注意看此处的ServerBuilder().是什么意思呢,在kotlin中,classname(). 常用于高阶函数中,作为高阶函数的参数。
形如:action: (Builder.() -> Int)
表示的是Function literals with receiver:这是一个方法,该方法不接收任何参数,该方法返回的是一个int,并且该方法是由Builder对象触发。

建造者设计模式 + 高阶函数 => DSL,设计模式,java,算法
其实按照如上的使用的时候,Idea给我们的提示就可以看出,通过将port方法的高阶函数定义为Builder.() -> Int,就相当于我们为port方法的上下文注入下this,而该this就是当前的Builder对象。

最终使用的时候如下

val server = Server.ServerBuilder()
        .port {
        8080
        }
        .address { 
            "www.baidu.com"
        }

借助了apply方法我们可以实现链式调用(因为port方法和address函数返回的都是builder对象),但是我们观察这种写法还是不够DSL化,为此我们给ServerBuilder添加一个构造方法

class ServerBuilder private constructor(){
        private var port: Int = 8080
        private var address: String = ""
        // 此处的this(), 表示次构造器要授权给主构造器
        constructor(init: ServerBuilder.() -> Unit): this() { init()}

        fun port(init: () -> Int) = apply { port = port2() }
        
        fun address(init: ServerBuilder.() -> String) = apply { address = init() }


        fun build(): Server = Server(this)
    }
val server = Server.ServerBuilder {

    }.port {
        8080
        }
        .address {
            "www.baidu.com"
        }
也可以写成如下,把port和address写入ServerBuilder的里面,因为ServerBuilder的里面可以拿到this上线文,故最终形态:
class Server private constructor(
    val port: Int,
    val address: String,
) {


    private constructor(builder: ServerBuilder): this(
        builder.port,
        builder.address
    )

    class ServerBuilder private constructor(){
        var port: Int = 8080
        var address: String = ""
        // 此处的this(), 表示次构造器要授权给主构造器
        constructor(init: ServerBuilder.() -> Unit): this() { init()}

        fun port(init: () -> Int) = apply { port = init() }

        fun address(init: ServerBuilder.() -> String) = apply { address = init() }

        fun build(): Server = Server(this)
    }
}

fun main() {
	val server = Server.ServerBuilder {
        port {
            8080
        }
        address {
            "www.baidu.com"
        }
    }.build()
}

为了进一步DSL化也为了向外界屏蔽ServerBuilder对象,我们可以给ServerBuilder添加静态方法

class Server private constructor(
    val port: Int,
    val address: String,
) {
    companion object {
       // inline fun build(block: ServerBuilder.() -> Unit) = Builder().apply(block).build()
       fun build(block: ServerBuilder.() -> Unit) = ServerBuilder {
            block()
       }.build()
    }


    private constructor(builder: ServerBuilder): this(
        builder.port,
        builder.address
    )

    class ServerBuilder private constructor(){
        var port: Int = 8080
        var address: String = ""
        // port方法的参数接收一个参数,该参数是一个lambda表达式(一个高阶函数),该port的方法的返回值是
        //apply的返回值,apply的返回值就是当前的ServerBuilder对象,完成的写法应该是this.apply
        // apply的内部则是将lambda的表达式的返回值赋值给了该ServerBuilder对象的port成员


        // 此处的this(), 表示次构造器要授权给主构造器
        constructor(init: ServerBuilder.() -> Unit): this() { init()}

        fun port(init: () -> Int) = apply { port = init() }

        fun address(init: ServerBuilder.() -> String) = apply { address = init() }

        fun build(): Server = Server(this)
    }
}
//测试
fun main() {
	val server = Server.build {
        port {
            8082
        }
        address {
            "www.baidu.com"
        }
    }
}

参考

https://stackoverflow.com/questions/44427382/what-does-mean-in-kotlin文章来源地址https://www.toymoban.com/news/detail-619656.html

到了这里,关于建造者设计模式 + 高阶函数 => DSL的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Java设计模式之一:建造者模式

    目录 一、什么是建造者模式 二、建造者模式如何使用 三、建造者模式的优势和应用场景 Java建造者模式是一种创建对象的设计模式,它通过将对象的构造过程分离出来,使得同样的构建过程可以创建不同的表示。建造者模式适用于创建复杂对象,它将对象的构建过程分解成

    2024年02月13日
    浏览(53)
  • Java与设计模式(6):建造者模式

    建造者模式(Builder Pattern)是一种创建型设计模式,它允许你逐步构造复杂对象。该模式将对象的构建过程与其表示分离,使得同样的构建过程可以创建不同的表示。 建造者模式的核心思想是将一个复杂对象的构建过程分解为多个简单的步骤,通过一系列的步骤来逐步构建对

    2024年02月11日
    浏览(39)
  • Java特性之设计模式【建造者模式】

    概述 建造者模式(Builder Pattern)使用多个简单的对象一步一步构建成一个复杂的对象。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式 一个 Builder 类会一步一步构造最终的对象。该 Builder 类是独立于其他对象的 主要解决 :主要解决在软件系统中,有

    2024年02月07日
    浏览(43)
  • Java建造者设计模式

    建造者模式(Builder Pattern)使用多个简单的对象一步一步构建成一个复杂的对象。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。一个 Builder 类会一步一步构造最终的对象。该 Builder 类是独立于其他对象的。 Builder(抽象建造者) :它为创建一个产

    2024年02月15日
    浏览(35)
  • JAVA设计模式第九章:建造者模式

    通过将对象的构建过程从表现层隔离出来,使得相同的构建过程可以用来创建不同的表现形式 用于简化和优化复杂对象的创建过程,提高创建效率和代码可读性; Product(产品角色): 要创建的产品对象 Builder(抽象建造者): 创建产品以及部件的接口定义 ConcreateBuilder(具体建造者): 抽

    2024年02月19日
    浏览(39)
  • 【Java设计模式】建造者模式 & 注解@Builder

    将一个复杂对象的构造与它的表示分离 ,使同样的构建过程可以创建不同的表示。它使将一个复杂的对象分解成多个简单的对象,然后一步步构建而成。 每一个具体建造者都相对独立,而与其它的具体建造者无关,因此可以很方便地替换具体建造者或增加新的具体建造者,

    2024年02月14日
    浏览(43)
  • 【Java 设计模式】创建型之建造者模式

    在软件开发中,建造者模式是一种创建型设计模式, 它将一个复杂对象的构建与其表示分离,使得同样的构建过程可以创建不同的表示 。建造者模式通常包括一个指导者(Director)类和多个建造者(Builder)类,指导者负责组织建造者的构建过程,而建造者负责具体的构建步

    2024年01月21日
    浏览(56)
  • 设计模式-建造者模式在Java中使用示例

    复杂对象的组装与创建 没有人买车会只买一个轮胎或者方向盘,大家买的都是一辆包含轮胎、方向盘和发动机等多个部件的完整汽车。 如何将这些部件组装成一辆完整的汽车并返回给用户,这是建造者模式需要解决的问题。 建造者模式又称为生成器模式,它是一种较为复杂

    2024年02月16日
    浏览(51)
  • Java设计模式之建造者模式详解(Builder Pattern)

    在日常的开发工作中,我们常常需要创建一些复杂的对象。这些对象可能包含许多不同的属性,并且这些属性的初始化过程可能相当复杂。在这种情况下,建造者模式是一种非常有用的设计模式,因为它允许我们分步骤地创建复杂的对象。 概念和原理: 建造者模式(Builder

    2024年02月09日
    浏览(45)
  • 【java设计模式】创建型模式介绍(工厂模式、抽象工厂模式、单例模式、建造者模式、原型模式)

    简介 本文介绍Java设计模式中创建型模式的五种 一、工厂模式 工厂模式(Factory Pattern)是 Java 中最常用的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。 工厂模式提供了一种将对象的实例化过程封装在工厂类中的方式。通过使用工

    2024年02月16日
    浏览(47)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包