scala中的单例对象和伴生类

40 阅读3分钟

主要内容:

1.单例对象

2.伴生类和伴生对象

3.应用-单例模式

(一)单例对象

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

格式

object 对象名{

属性

方法

}

代码如下:

package level02

object class04 {
  /*
  * 伴生类 和 半生对象
  * 实现 单例模式: 一个类只能有一个对象
  */

  //让构造函数私有,在类的外部,就无法使用new创造新的对象
  class Student private () {

    //private修饰的属性,在类的外部(当前的class之外)无法访问!
    //private val name:String = "小花"
    val name:String = "小花"

  }
  //在伴生对象中创建类实例,并返回
  object Student {
    val instance = new Student() //在伴生类中,可以访问伴生对象的私有成员
    def getInstance:Student = {
      instance
    }
  }

  def main(args: Array[String]): Unit = {
    val stu1 = Student.getInstance
    val stu2 = Student.getInstance
  }
}

注意:

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

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

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

(二)伴生类和伴生对象

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

package level02

object class01 {
  /*
  *  伴生类和伴生对象
  * 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是错误的,无法访问
    Student.sayHello(stu1)
  }

}

结果如下: 屏幕截图 2025-11-04 083719.png

代码说明:

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

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

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

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

(三)应用-单例模式

【讲解】 什么是单例模式?

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

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

代码如下:

package level02

object class05 {
  /*
    * apply
    *  apply方法写在2伴生对象中,可以在创建对象的时候,省略new关键字
    *  实现单例模式
    *  在伴生对象中,创建一个对象,在apply方法中返回它
     */

  class Person() {
    println("主购造器被调用..........")
  }

  object Person {
    val p = new Person
    def apply():Person={
      p
    }
  }

  def main(args: Array[String]): Unit = {
    //    val p1=Person


    val p1 = Person() // 自动调用apply
    val p2 = Person()

    println(p1)
    println(p2)
    println(p1 == p2)
  }


}

结果如下: 屏幕截图 2025-11-04 084619.png