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. presto + slider 提交计算至yarn

    10.112.28.240 prestocli 10.183.225.158 perstoser hive-site.xml useUnicode=true&characterEncoding ...

  2. 20145313张雪纯 《Java程序设计》8周学习总结

    20145313张雪纯 <Java程序设计>8周学习总结 教材学习内容总结 java.util.logging包的优点在于提供了日志功能相关类与接口,不必额外配置日志组件就可以在标准jav ...

  3. spi nor flash使用汇总

    Overview SPI flash, 分为spi flash, DUAL spi flash, QUAD spi flash, 3-wire spi, 4-wire spi, 6-wire spi. ...

  4. 比较字符串CompareTo的用法及注意

    CompareTo用法 static void Main(string[] args)         {             string str = "1";        ...

  5. svn常用维护命令

    公司版本管理同时用的svn和gitlab,有互补作用 这边写一写慢慢积累自己用过的svn常用维护 查看版本范围之间的变化: [root@192-168-2-82 mnt]# svn log -r 45 ...

  6. Linux用户及权限分配

    一.用户分类 所有者 u; 所属组 g; 其它用户 o; 所有用户 a; 二.用户管理 //查看用户 id user //添加用户 useradd user //设置密码 passwd user // ...

  7. codeforce AIM tech Round 4 div 2 B rectangles

    2017-08-25 15:32:14 writer:pprp 题目: B. Rectangles time limit per test 1 second memory limit per test ...

  8. 测试Python类成员的单下划线,双下划线,两头下划线的区别

    首先原谅一个菜鸟叫他“两头下划线”.记得在windows编程中,很多宏定义使用下划线+大写,给人逼格很高的错觉.对于Python下划线的认识,大概是从__dict__这个属性开始的,看__dict__ ...

  9. ubuntu18.04里更新系统源和pip源

    一.修改ubuntu系统源 我的ubuntu系统是在清华的开源网站上下的,所以我还以为他应该就帮我弄好源了,可是没想到下载的还是非常慢,看到下载的时候网址前还有个us,就知道不是国内源了.所以这里我们 ...

  10. 【小而优】 如何实现 tail -f 动态显示日志时高亮显示关键字

    前言 如果你在linux下工作,那用tail -f跟踪一个日志文件的输出内容应该是家常便饭了. 但是,有时你更关心的是一些敏感字词,希望能够在动态跟踪的同时,把这些字词高亮出来,比如日志中的 ERRO ...