当前位置: 首页 > news >正文

【Spinalhdl】Scala编程之伴生对象

【Spinalhdl】Scala编程之伴生对象

    • 不使用关键字new创建新实例
      • 启用功能
      • 创建多个构造函数
    • 添加unapply方法
      • unapply 返回不同类型
      • unapply 提取器
    • 要点

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

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

启用功能

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

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

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

  • 从命令行启动Scala REPL(使用如下命令)
  • scala 输入并按[Enter]键 :paste
  • 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] = Nonevar age: Option[Int] = Noneoverride def toString = s"$name, $age"
}
object Person {// a one-arg constructordef apply(name: Option[String]): Person = {var p = new Personp.name = namep}// a two-arg constructordef apply(name: Option[String], age: Option[Int]): Person = {var p = new Personp.name = namep.age = agep}
}

如果像之前那样将代码粘贴到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方法允许你将一个类的实例解构为它的各个组件

http://www.mrgr.cn/news/28007.html

相关文章:

  • 在linux里如何利用vim对比两个文档不同的行数
  • 23种设计模式-观察者(Observer)设计模式
  • 【Linux网络编程】简单的UDP网络程序
  • 自动化生成测试用例:利用OpenAI提升电商网站测试覆盖率
  • 自定义call方法和apply方法
  • 修改mysql默认字符集
  • 内存dump文件分析
  • MS SQL Server 实战 排查多列之间的值是否重复
  • linux进程间通信——学习与应用命名管道, 日志程序的使用与实现
  • 全方位解读信息架构:从挑战到解决方案,推动企业数字化转型的全面指南
  • 【详细原理】蒙特卡洛树搜索
  • 【机器学习-监督学习】朴素贝叶斯
  • k8s的NodeIP、PodIP、ClusterIP、ExternalIP
  • 【RabbitMQ】工作模式
  • 一次RPC调用过程是怎么样的?
  • Spark实操学习
  • C++学习笔记(26)
  • 电子电气架构---智能汽车应该是怎么样的架构?
  • [C++]类和对象(上)
  • Java接口的艺术:探索接口特性与面向接口编程
  • Linux 基本使用和 web 程序部署 ( 8000 字 Linux 入门 )
  • JVM JMM 专题篇 ( 12000 字详解 )
  • 基于SSM的在线家教管理系统(含源码+sql+视频导入教程+文档+PPT)
  • WPF颜色(SolidColorBrush)和Win32颜色(COLOREF)互转的方法
  • 详解QT元对象系统用法
  • 「漏洞复现」用友U8 Cloud AddTaskDataRightAction SQL注入漏洞