Scala中的单例对象和伴生类

37 阅读3分钟

内容:

1. 伴生类的定义;

2. 伴生类的作用和格式;

3. 单例对象的定义;

  1. 单例对象的作用和格式。

前面我们学习了如何定义类和对象,类是对象的抽象,对象是属性的集合。在

一般情况下,我们都是通过new Class的方式来创建对象,但是,有些时候,我们希望简单一些,直接去创建对象。

单例对象

用 object 关键字来创建一个单例对象。单例对象在整个应用程序中只有一个实例,适合用于存放工具方法、常量或共享状态。

说明

1. 通过object关键字创建的是一个对象,不是一个类型。

2. 不能使用new关键字:声明单例对象的时候不可以使用new关键字。

3. 不能传递参数:单例对象无法传递参数。

伴生类和伴生对象

定义

当同名的类和单例对象在同一个源码文件时,这个类称为单例对象的伴生类,对象称为类的伴生对象。

代码说明:

1. 类名和对象名必须同名。

2. 必须在同一个源码文件中。

package level02

object class001 {
  /*
  * 伴生类 和 伴生对象
  * 1,类和对象的名字是一样的
  * 2,他们在同一个文件中
  * 特点:可以访问对方的私有(private)树形
   */
  class Student() {
    // private 修饰的属性,在类的外部(在当前的class之外)无法访问
    private val name:String = "小花"
  }
  object Student {
    def sayHello(stu:Student):Unit = {
      // 在伴生对象中,访问伴生类的私有属性
      println(s"${stu.name}")
    }
  }

  def main(args: Array[String]): Unit = {
    val stu1 = new Student()
    // stu1.name 是错误的,无法访问
    Student.sayHello(stu1)
  }
}

特点:Scala的伴生对象和伴生类的特点:

伴生对象和类的私有private成员可以相互访问。

[编码示例] :访问私有方法

package level02

object class002 {
  /*
  * 伴生类 和 伴生对象 实现 单例模式:一个类只能有一个对象
   */
  // 1,让构造函数变成私有的,在类的外部,就无法通过new来创造对象了
  class Student private () {
    // private 修饰的属性,在类的外部(在当前的class之外)无法访问
    private val name:String = "小花"
  }
  //2,在伴生对象在中创建类的实例,并返回
  object Student {
    val instance = new Student()//在伴生对象中,可以访问伴生类的私有成员
    def getInstance:Student = {
      instance
    }
  }

  def main(args: Array[String]): Unit = {
    // val stu1 = new Student()
    // stu1.name 是错误的,无法访问
    val stu1 = Student.getInstance
    val stu2 = Student.getInstance
    println(stu1==stu2)//true
  }
}

应用-单例模式

什么是单例模式?

就是通过技术手段,让某个类只能有一个对象实例。这样做的好处就节约资源,能更加精准地管理。

思路:使用private来修饰构造器,这样在类的外部就无法访问了。在伴生对象中提供获取这个实例的入口方法。

示例代码

class Logger private () {

  def log(message: String): Unit = {

    println(message)

  }
}

object Logger {

  private val instance = new Logger()

  def getInstance: Logger = instance

}

object Single2{

  def main(args: Array[String]): Unit = {

    // 使用伴生对象获取 Logger 实例

    val logger1 = Logger.getInstance  
    val logger2 = Logger.getInstance  
    println(logger1 == logger2)

    logger.log("This is a log message.")

  }
}