1. 单例对象

Scala没有静态方法或静态字段,可以使用object来达到这个目的,对象定义了某个类的单个实例:

object Account{
  private var lastNumber = 0
  def newUniqueNumber () = {lastNumber += 1; lastNumber}
}

当你在应用程序中需要一个新的唯一账号时,调用Account.newUniqueNumber()即可.对象的构造器在该对象第一次被使用时调用.在本例中,Account的构造器在Account.newUniqueNumber的首次调用时执行.如果一个对象从未被使用,那么起构造器也不会被执行.

对象本质上可以拥有类的所有特性,但是不能提供构造器参数.

在Scala中可以用对象来实现:

  • 作为存放工具函数或常亮的地方
  • 高效的共享单个不可变实例
  • 需要用单个实例来协调某个服务时(参考单例模式)

2. 伴生对象

在Java中,通常会用到既有实例方法又有静态方法的类,在Scala中,可以通过类和类同名的"伴生"对象来达到同样的目的:

class Account{
  val id = Account.newUniqueNumber()
  private var balance = 0.0
  def deposit(amount : Double) { balance += amount }
  ...
}

// 伴生对象
object Account{
  private var lastNumber = 0
  def newUniqueNumber () = {lastNumber += 1; lastNumber}
}

类和它的伴生对象可以相互访问私有特性.它们必须在同一个源文件中.

3. apply方法

我们通常会定义和使用对象的apply方法.当遇到如下形式的表达式时,apply方法就会被调用:

Object(参数1,参数2,...,参数N)

通常,这样一个apply方法返回的是伴生类的对象. 举例来说,Array对象定义了apply方法,让我们可以用下面这样的表达式来创建数组:

Array("Mary", "had", "a", "little", "lamb")

不使用构造器,而是使用apply方法,对于使用嵌套表达式而言,省去new关键字会方便很多:

Array(Array(1,7), Array(2,9))

下面有一个定义apply方法的示例:

class Account private (val id :Int, initialBalance: Double){
  private var balance = initialBalance
  ...
}

// 伴生对象
object Account{
  def apply(initialBalance : Double){
    new Account(newUniqueNumber(), initialBalance)
  }
  ...
}

这样我们就可以使用如下方式创建账号了:

val acct = Account(1000.0)

4. 应用程序对象

每个Scala程序都必须从一个对象的main方法开始,这个方法的类型为Array[String]=>Unit:

object Hello{
  def main(args: Array[String]){
    println("Hello world!")
  }
}

除了每次都提供自己main方法外,你可以扩展App特质,然后将程序代码放入构造器方法体内:

object Hello extends App{
  println("Hello world!")
}

如果需要命令行参数,则可以通过args属性得到:

object Hello extends App{
  if(args.length > 0){
    println("Hello, " + args(0))
  }
  else{
    println("Hello world!")
  }
}

5. 枚举

不同于Java,Scala中没有枚举类型,需要我们通过标准库类Enumeration来实现:

object BusinessType extends Enumeration{
  var FLIGHT, HOTEL, TRAIN, COACH = Value
}

继承Enumeration类,实现一个BusinessType对象,并以Value方法调用初始化枚举中的所有可选值.在这里我们定义了4个业务线类型,然后用Value调用它们初始化.

每次调用Value方法都返回内部类的新实例,该内部类也叫做Value.或者,可以向Value方法传入ID,名称:

val FLIGHT = Value(0, "FLIGHT")
val HOTEL = Value(10) // 名称为"HOTEL"
val TRAIN = Value("TRAIN") // ID为11

如果不指定ID,ID为上一个枚举值上加一,如果不指定名称,名称默认为字段名.定义完成后,可以使用BusinessType.FLIGHT,BusinessType.HOTEL,BusinessType.TRAIN等来引用:

def businessHandle(business: BusinessType.Value): Unit ={
  if(business == BusinessType.FLIGHT){
    println("this is a flight behavior")
  }
  else if(business == BusinessType.HOTEL){
    println("this ia a hotel behavior")
  }
}

def main(args: Array[String]): Unit = {
  val business = BusinessType.FLIGHT
  businessHandle(business) // this is a flight behavior
}

如果觉的BusinessType.FLIGHT比较冗长繁琐,可以使用如下方式引入枚举值:

import BusinessType._

使用时直接使用枚举值名称即可:

def businessHandle(business: BusinessType.Value): Unit ={
  if(business == FLIGHT){
    println("this is a flight behavior")
  }
  else if(business == HOTEL){
    println("this ia a hotel behavior")
  }
}

记住枚举值的类型是BusinessType.Value而不是BusinessType,后者是拥有这些值的对象,可以增加一个类型别名:

object BusinessType extends Enumeration{
  type BusinessType = Value
  var FLIGHT, HOTEL, TRAIN, COACH = Value
}

如下使用:

def businessHandle(business: BusinessType): Unit ={
  if(business == FLIGHT){
    println("this is a flight behavior")
  }
  else if(business == HOTEL){
    println("this ia a hotel behavior")
  }
}

枚举值的ID可以通过id方法返回,名称通过toString方法返回:

val business = FLIGHT
println("ID:" + business.id + "   name:" + business.toString) // ID:0   name:FLIGHT

可以通过如下方式输出所有的枚举值:

for(business <- BusinessType.values){
  println("ID:" + business.id + "   name:" + business.toString)
}

ID:0   name:FLIGHT
ID:1   name:HOTEL
ID:2   name:TRAIN
ID:3   name:COACH

你也可以通过枚举的ID或名称来进行查找定位:

val FLIGHT1 = BusinessType(0)
println("ID:" + FLIGHT1.id + "   name:" + FLIGHT1.toString)
val FLIGHT2 = BusinessType.withName("FLIGHT")
println("ID:" + FLIGHT2.id + "   name:" + FLIGHT2.toString)

ID:0   name:FLIGHT
ID:0   name:FLIGHT    

[Scala]Scala学习笔记五 Object的更多相关文章

  1. Scala入门学习笔记三--数组使用

    前言 本篇主要讲Scala的Array.BufferArray.List,更多教程请参考:Scala教程 本篇知识点概括 若长度固定则使用Array,若长度可能有 变化则使用ArrayBuffer 提 ...

  2. Learning ROS for Robotics Programming Second Edition学习笔记(五) indigo computer vision

    中文译著已经出版,详情请参考:http://blog.csdn.net/ZhangRelay/article/category/6506865 Learning ROS for Robotics Pr ...

  3. go微服务框架kratos学习笔记五(kratos 配置中心 paladin config sdk [断剑重铸之日,骑士归来之时])

    目录 go微服务框架kratos学习笔记五(kratos 配置中心 paladin config sdk [断剑重铸之日,骑士归来之时]) 静态配置 flag注入 在线热加载配置 远程配置中心 go微 ...

  4. C#可扩展编程之MEF学习笔记(五):MEF高级进阶

    好久没有写博客了,今天抽空继续写MEF系列的文章.有园友提出这种系列的文章要做个目录,看起来方便,所以就抽空做了一个,放到每篇文章的最后. 前面四篇讲了MEF的基础知识,学完了前四篇,MEF中比较常用 ...

  5. (转)Qt Model/View 学习笔记 (五)——View 类

    Qt Model/View 学习笔记 (五) View 类 概念 在model/view架构中,view从model中获得数据项然后显示给用户.数据显示的方式不必与model提供的表示方式相同,可以与 ...

  6. java之jvm学习笔记五(实践写自己的类装载器)

    java之jvm学习笔记五(实践写自己的类装载器) 课程源码:http://download.csdn.net/detail/yfqnihao/4866501 前面第三和第四节我们一直在强调一句话,类 ...

  7. Typescript 学习笔记五:类

    中文网:https://www.tslang.cn/ 官网:http://www.typescriptlang.org/ 目录: Typescript 学习笔记一:介绍.安装.编译 Typescrip ...

  8. ES6学习笔记<五> Module的操作——import、export、as

    import export 这两个家伙对应的就是es6自己的 module功能. 我们之前写的Javascript一直都没有模块化的体系,无法将一个庞大的js工程拆分成一个个功能相对独立但相互依赖的小 ...

  9. muduo网络库学习笔记(五) 链接器Connector与监听器Acceptor

    目录 muduo网络库学习笔记(五) 链接器Connector与监听器Acceptor Connector 系统函数connect 处理非阻塞connect的步骤: Connetor时序图 Accep ...

随机推荐

  1. spark client + yarn计算

    前提:完成hadoop + kerberos安全环境搭建. 安装配置spark client: 1. wget https://d3kbcqa49mib13.cloudfront.net/spark- ...

  2. 重写(overwrite)、重载(overload)和覆盖(override)三者之间的区别

    覆盖:子类继承了父类的同名无参函数.当子类从父类继承了一个无参函数,而又定义了一个同样的无参函数,则子类定义的方法覆盖父类的方法,称为覆盖. 重载:子类继承了父类的同名有参函数.当子类继承了父类的一个 ...

  3. Python-自省机制

     help 如果说能够通过一个函数就能够学会 Python,那这个函数一定就是 Python 提供的第一 个自带说明 help().help 函数的作用就是查看对象的帮组文档.比如: >> ...

  4. terminal配置

    阅读目录 前言 使用 tmux 复用控制台窗口 在命令行中快速移动光标 在命令行中快速删除文本 快速查看和搜索历史命令 快速引用和修饰历史命令 录制屏幕并转换为 gif 动画图片 总结 回到顶部 前言 ...

  5. Show Desktop Pro FAQ

    Q. Will the desktop background image be restored after quit? A: Yes. Right now, "Hide icons&quo ...

  6. CSS控制滚动条的样式

    到今天(2018年10月25日)为止, 这还是chrome上的一个实验性特性: ::-webkit-scrollbar{width:4px;height:4px;} ::-webkit-scrollb ...

  7. heartbeat 编译安装配置

    一.heartbeat介绍 heartbeat是HA高可用集群的一个重要组件,heartbeat实现了资源转移和心跳信息传递.它的常用组合方式为heartbeat v1,heartbeat v2+cr ...

  8. spring boot将jar包转换成war包发布

    spring boot将jar包转换成war包发布步骤 将<packaging>jar</packaging>修改为<packaging>war</packa ...

  9. Python学习札记(二十一) 函数式编程2 map/reduce

    参考:map/reduce Note 1.map():map()函数接收两个参数,一个是函数,一个是Iterable.map将传入的函数依次作用到序列的每个元素,并把结果作为新的Iterator返回. ...

  10. Maven打包命令

    mvn clean 会把原来target目录给删掉重新生成.mvn install 安装当前工程的输出文件到本地仓库,然后打包mvn clean install 先删除target文件夹 ,然后打包到 ...