【SpinalHDL】Scala编程之伴生对象

Scala中的伴生对象是指和在同一个文件中声明的,并且和类同名的对象。例如,下面的代码保存在名为Pizza的文件中。在Scala中,这个object Pizza 对象被认为是class Pizza 类的伴生对象:

class Pizza {
}
object Pizza {
}

这样设计的优势是伴生对象及其类可以访问彼此的私有成员(字段方法),这意味着这个类中的printFilename 方法可以工作,因为它可以访问它的伴生对象中的HiddenFilename字段:

class SomeClass {
    def printFilename() = {
        println(SomeClass.HiddenFilename)
    }
}
object SomeClass {
    private val HiddenFilename = "/tmp/foo.bar"
}

伴生对象提供的功能远不止这些,我们将在本课剩下的内容中演示它的几个最重要的功能。

不使用关键字new创建新实例

在一些Spinalhdl示例中创建某些类的新实例时不必在类名之前使用关键字new,如下面的例子所示:

val zenMasters = List(
    Person("Nansen"),
    Person("Joshu")
)

这个功能来自于伴生对象的使用。当伴生对象中定义一个方法时,它对Scala编译器有特殊的意义。Scala内置了一些语法糖,可以让你输入这样的代码: apply

val p = Person("Fred Flinstone")

在编译过程中,编译器将该代码转换为以下代码:

val p = Person.apply("Fred Flinstone")

伴生对象中的apply 方法就像一个工厂方法,Scala的语法糖允许你使用上面的语法,不用关键字就能创建新new的类实例:

启用功

为了演示这个功能是如何工作的,下面是一个Person类名和它的伴生对象中的apply方法:

class Person {
    var name = ""
}
object Person {
    def apply(name: String): Person = {
        var p = new Person
        p.name = name
        p
    }
}

为了测试这段代码,可以使用下面的技巧将类和对象同时粘贴到Scala REPL中:

  • 从命令行启动Scala REPL(使用如下命令)

  • scala 输入paste 并按[Enter]键 :

  • REPL的响应应该是这样的:

    // Entering paste mode (ctrl-D to finish)

  • 现在将类和对象同时粘贴到REPL中

  • 按Ctrl-D完成"paste"过程
    当这个过程正常运行时,你应该在REPL中看到如下输出:

    defined class Person
    defined object Person

REPL要求使用这种技术同时输入一个类和它的伴生对象。

现在可以像这样创建一个类的新实例: Person

val p = Person.apply("Fred Flinstone")

该代码直接调用伴生对象。更重要的是还可以像这样创建一个新实例: apply

val p = Person("Fred Flinstone")

更加便利使用此功能如下

val zenMasters = List(
    Person("Nansen"),
    Person("Joshu")
)

需要说明的是,在这个过程中发生的是:

  • 输入 val p = Person("Fred")
  • Scala编译器会发现 new Person之前没有关键字
  • 编译器会在类的伴生对象中查找与输入的类型签名匹配的方法 apply Person
  • 如果找到方法就使用它 apply,如果没有会得到一个编译错误

创建多个构造函数

可以在一个伴生对象中创建多个方法来提供多个构造函数。下面的代码展示了如何创建一个和两个参数的构造函数。

class Person {
    var name: Option[String] = None
    var age: Option[Int] = None
    override def toString = s"$name, $age"
}
object Person {
    // a one-arg constructor
    def apply(name: Option[String]): Person = {
        var p = new Person
        p.name = name
        p
    }
    // a two-arg constructor
    def apply(name: Option[String], age: Option[Int]): Person = {
        var p = new Person
        p.name = name
        p.age = age
        p
    }
}

如果像之前那样将代码粘贴到REPL中,你会看到可以像下面这样创建新实例: Person

val p1 = Person(Some("Fred"))
val p2 = Person(None)
val p3 = Person(Some("Wilma"), Some(33))
val p4 = Person(Some("Wilma"), None)

打印这些值,会得到如下结果:

val p1: Person = Some(Fred), None
val p2: Person = None, None
val p3: Person = Some(Wilma), Some(33)
val p4: Person = Some(Wilma), None

运行这样的测试时最好清除REPL的内存 :reset:paste

添加unapply方法

就像在伴生对象中添加apply方法可以构造新的对象实例一样,添加unapply方法可以反构造对象实例。我们将用一个例子来演示:

下面是一个Person类及其伴生对象的不同版本:

class Person(var name: String, var age: Int)
object Person {
    def unapply(p: Person): String = s"${p.name}, ${p.age}"
}

注意伴生对象定义了一个unapply方法。这个方法接受Person类型为的输入参数,并返回String。要手动测试该unapply方法首先创建一个新Person实例:

val p = new Person("Lori", 29)

然后像这样unapply测试:

val result = Person.unapply(p)

这是REPL中的unapply结果:

scala> val result = Person.unapply(p)
result: String = Lori, 29

如上所示对给定的实例进行解构。

在Scala中把一个unapply方法放在伴生对象中时,我们就说你创建了一个提取器方法,因为你已经创建了一种从Person对象中提取字段的unapply方法:

unapply 返回不同类型

在上面例子中unapply返回String a,但也可以让它返回任何东西。下面的示例返回元组中的两个字段:

class Person(var name: String, var age: Int)
object Person {
    def unapply(p: Person): Tuple2[String, Int] = (p.name, p.age)
}

这个方法在REPL中如下所示:

scala> val result = Person.unapply(p)
result: (String, Int) = (Lori,29)

因为这个unapply方法以元组的形式返回类字段,所以你也可以这样做:

scala> val (name, age) = Person.unapply(p)
name: String = Lori
age: Int = 29

unapply 提取器

创建提取器的一个好处是,如果你遵循正确的Scala约定,它们会在匹配表达式中启用一种方便的模式匹配形式

要点

  • 伴生对象是与A在同一个文件中声明的,并且与类 objectclass同名
  • 伴生对象及其类可以访问彼此的私有成员
  • 伴生对象的apply方法让你无需使用关键字 new就能创建类的新实例
  • 伴生对象的unapply方法允许你将一个类的实例解构为它的各个组件
相关推荐
禁默8 分钟前
深入浅出:AWT的基本组件及其应用
java·开发语言·界面编程
Code哈哈笑17 分钟前
【Java 学习】深度剖析Java多态:从向上转型到向下转型,解锁动态绑定的奥秘,让代码更优雅灵活
java·开发语言·学习
程序猿进阶21 分钟前
深入解析 Spring WebFlux:原理与应用
java·开发语言·后端·spring·面试·架构·springboot
qq_4336184423 分钟前
shell 编程(二)
开发语言·bash·shell
charlie11451419137 分钟前
C++ STL CookBook
开发语言·c++·stl·c++20
袁袁袁袁满38 分钟前
100天精通Python(爬虫篇)——第113天:‌爬虫基础模块之urllib详细教程大全
开发语言·爬虫·python·网络爬虫·爬虫实战·urllib·urllib模块教程
ELI_He99944 分钟前
PHP中替换某个包或某个类
开发语言·php
m0_748236111 小时前
Calcite Web 项目常见问题解决方案
开发语言·前端·rust
倔强的石头1061 小时前
【C++指南】类和对象(九):内部类
开发语言·c++
Watermelo6171 小时前
详解js柯里化原理及用法,探究柯里化在Redux Selector 的场景模拟、构建复杂的数据流管道、优化深度嵌套函数中的精妙应用
开发语言·前端·javascript·算法·数据挖掘·数据分析·ecmascript