Scala 面向对象

该博客主要介绍了Scala中类的相关知识,包括类的定义和使用、构造器、继承、重写、抽象类、伴生类与伴生对象、case class和Trait。其中提到占位符使用需指定类型,构造器执行顺序,继承用extend,Trait类似Java接口可实现多重继承等内容。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

目录

 

概述:

1.类的定义和使用

2、构造器

3、继承

4、重写

5.抽象类

6.伴生类与伴生对象

7.case class

8. Trait


概述:

1.类的定义和使用

package ScalaTest1

object SimpleObjectApp {
  def main(args: Array[String]): Unit = {
    val person = new People()
    person.name = "Messi"
    println(person.name + "..." + person.age)

    person.printInfo()
    // person(person.gender) 调用不到,有private [this]修饰

    println("invoke eat method: " + person.eat())
    person.watchFootball("Barcelona")
  }
}

class People{
  // 定义属性
  var name:String = _     //使用占位符代替  var name:String = ""
  val age:Int = 10

  private [this] val gender = "male"   // 使用private [this] 只能在class内部被访问,

  def printInfo(): Unit ={
    println("gender: " + gender)
  }
  def eat():String = {
    name + "eat..."
  }
  def watchFootball(teamName: String): Unit = {
    println(name + " is watching match of "+ teamName)
  }
}

 

补充

占位符变量在不知道赋什么值的时候可以先用占位符

注意:使用占位符必须先指定类型

 

2、构造器

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

    // 传两个参数,对应到主构造器
    val person = new Person("zhansan", 32)
    println(person.name + ":" + person.age + ":" + person.school)

    // 传三位参数,对应到附属构造器
    val person2 = new Person("zhansi", 18, "M")
    println(person2.name + ":" + person2.age + ":" + person2.school + ":" + person2.gender)

  }
}


// 跟在类名后面的称为“主构造器”
class Person(val name: String, val age: Int) {
  println("Person Constructor enter...")

  val school         = "ustc"
  var gender: String = _

  // 附属构造器,把主构造器的参数直接拿来用
  def this(name: String, age: Int, gender: String) {
    this(name, age) // 附属构造器的第一行,必须调用主构造器,或者其它附属构造器
    this.gender = gender

  }

  println("Person Constructor leave...")
}

运行结果可以看到先执行了主构造器,再执行附属构造器

Person Constructor enter...
Person Constructor leave...
zhansan:32:ustc

Person Constructor enter...
Person Constructor leave...
zhansi:18:ustc:M

3、继承

使用extend 继承父类

// Student 继承了Person,如果我们要使用Person的属性,Student构造器的参数不需要加 val 或var,否则要加上
class Student(name:String, age:Int, var major:String) extends Person(name, age){
  println("Student Constructor enter...")

  println("Student Constructor leave...")
}

调用student

val student = new Student("zhangwu", 16, "Math")
println(student.name + ":" + student.major)
println(student)

输出结果:先执行父类主构造器,再执行Student的构造器

Person Constructor enter...
Person Constructor leave...

Student Constructor enter...
Student Constructor leave...
zhangwu:Math

com.hhcycj.scala.object_04.Student@39aeed2f

println(student) 相当于 println(student.toString)

4、重写

通过 override 关键字,重写 Object 的 toString 方法

override def toString: String = "override def toString"

重写父类的 school 属性

 override val school: String = "li yang xiao xue"

5.抽象类

package ScalaTest1

object AbstractApp {

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

    val student=new Students2()
    println(student.name)
    student.speak

  }

}

/**
  * 类的一个或者多个方法没有完整的实现(只有定义,没有实现)
  */

abstract class Person2 {

  def speak

  val name: String
  val age: Int


}

// 子类实现抽象类
class Students2 extends Person2 {
  override def speak: Unit = {
    println("speak")
  }

  override val name: String = "lisi"
  override val age: Int = 20
}

6.伴生类与伴生对象

package ScalaTest1

object ApplyApp {

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

    for (i <- 1 to 10) {
      ApplyTest.incr
    }
    println(ApplyTest.count)

    val b = ApplyTest() // ==> Object.apply

    println("==================")

    // new一个class对象
    val c = new ApplyTest()
    println(c)
    c()

    // 类名() ==> Object.apply
    // 对象() ==> Class.apply
  }

}

/**
  * 伴生类与伴生对象
  * 若有一个class和其同名的object
  * 则就称object是class的伴生对象,class是object的伴生类
  */

class ApplyTest {

  def apply() = {
    println("Class ApplyTest apply ....")

  }

}

object ApplyTest {

  println("Object ApplyTest entering....")
  var count = 0

  def incr = {
    count = count + 1
  }

  // 最佳实践:在Object的apply方法中去new class
  def apply() = {
    println("Object ApplyTest apply ....")

    // 在Object中new一个class
    new ApplyTest
  }

  println("Object ApplyTest leaving....")
}

运行结果:

Object ApplyTest entering....
Object ApplyTest leaving....
10
Object ApplyTest apply ....
==================
com.lihaogn.ApplyTest@75bd9247
Class ApplyTest apply ....

7.case class

package ScalaTest1

object CaseClassApp {

  def main(args: Array[String]): Unit = {
    println(Dog("wangwang").name)
  }

}

// case class 不用new
// 通常用在模式匹配里
case class Dog(name: String)

8. Trait

类似于Java中的接口,与接口不同的是,它还可以定义属性和方法的实现。

一般情况下Scala的类只能够继承单一父类,但是如果是 Trait(特征) 的话就可以继承多个,从结果来看就是实现了多重继承。

// 第一个Trait用extends,其他的用with
class class-name(xxx:xxx) extends Cloneable with Logging with ....

 

### Scala 面向对象编程概念 #### 类与对象 在Scala中,类是创建对象的蓝图。通过定义类来封装数据和操作这些数据的方法[^1]。 ```scala class Person(val name: String, val age: Int) { def introduce(): Unit = println(s"Name is $name and Age is $age") } ``` 上述代码展示了如何定义一个`Person`类,并为其提供了一个名为`introduce()`的方法用于自我介绍。 对于单例模式的支持,Scala提供了特殊的语法糖——`object`关键字用来声明单例对象。这使得编写工具函数变得非常方便。 ```scala object MathUtil { def add(a: Int, b: Int): Int = a + b } println(MathUtil.add(3, 5)) ``` 这里展示的是一个简单的加法计算工具类`MathUtil`作为单例存在。 #### 特质(Trait) 特质类似于其他语言里的接口,但是更加强大因为它可以包含具体实现。可以通过`with`关键字将多个特质组合在一起赋予某个特定类型的实例上[^4]。 ```scala trait Logger { def log(message: String): Unit = println(s"[LOG]:$message") } val userLogger = new AnyRef with Logger userLogger.log("User logged in.") ``` 这段代码说明了怎样利用特质为任意类型添加日志记录功能而无需修改原有结构。 #### 继承机制 当涉及到继承关系时,子类可以在其主构造器里显式调用超类(即父类)的构造器完成初始化工作;而在Java里面则是隐式的处理方式[^5]。 ```scala class Employee(name: String, age: Int, private var salary: Double) extends Person(name, age){ override def toString: String = super.toString + s"\nSalary=$salary" } val emp = new Employee("Alice", 27, 60000) emp.introduce() println(emp) ``` 此部分演示了员工类扩展自人员基类的过程以及覆盖toString方法来自定义字符串表示形式。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值