零、本节学习目标
- 掌握类的定义与实例化
- 理解单例对象和伴生对象
- 掌握构造器和辅助构造器
- 掌握抽象类和特质
一、类
(一)类的定义
- 对象是类的具体实例,类是抽象的,不占用内存,而对象是具体的,占用存储空间。
- 面向对象三大特性之一:封装(encapsulation) - 封装数据和操作
- Scala中一个简单的类定义是使用关键字class,类名首字母必须大写。类中的方法用关键字def定义
- 创建net.huawei.day04包,在包里创建User类,包含三个私有属性和一个公共方法
-
package net.zyf.day04
-
/**
-
* 功能:用户类
-
* 作者:zyf
-
* 日期:2023年03月日
-
*/
-
class User{
-
private var name = "张三丰"
-
private var gender = "男"
-
private var age = 25
-
def speak(): Unit = println("我叫" + name + "," + gender + ",今年" + age + "岁了~")
-
}
- 说明:在Scala里,如果一个类不写访问修饰符,那么默认访问级别为
public
,这与Java是不一样的。
(二)类的实例化
- 关键字
new
用于创建类的实例 - 实例化
User
类,调用其speak()
方法 - 在
net.huawei.day04
包里创建TestUser
对象
-
package net.zyf.day04
-
/**
-
* 功能:
-
* 作者:zyf
-
* 时间:2023年03月23号
-
*/
-
object TestUser {
-
def main(args: Array[String]): Unit = {
-
// 创建用户对象
-
val user = new User()
-
// 调用对象方法
-
user.speak()
-
}
-
}
- 运行查看结果
- 访问私有属性
name
,系统会报错
-
怎么才能访问类的私有属性呢?后面我们会定义
setter
和getter
来访问私有属性。
二、单例对象
(一)单例对象概念
- Scala中没有
静态方法
或静态字段
,但是可以使用关键字objec
t定义一个单例对象
,单例对象中的方法相当于Java中的静态方法
,可以直接使用“单例对象名.方法名
”方式进行调用。单例对象除了没有构造器参数外,可以拥有类的所有特性。
(二)案例演示
- 在
net.huawei.day04
里创建Person
单例对象,包含三个私有属性和一个公共方法
-
package net.zyf.day04
-
/**
-
* 功能:
-
* 作者:zyf
-
* 时间:2023年03月00号
-
*/
-
object Person {private var name = "陈燕文"
-
private var gender = "女"
-
private var age = 18
-
def speak(): Unit = println("我叫" + name + "," + gender + ",今年" + age + "岁了~")
-
}
- 在
net.huawei.day04
包里创建TestPerson
对象
-
package net.zyf.day04
-
/**
-
* 功能:
-
* 作者:zyf
-
* 时间:2023年03月00号
-
*/
-
object TestPerson {
-
def main(args: Array[String]): Unit = {
-
//直接调用对象方法()
-
Person.speak()
-
}
-
}
三、伴生对象
(一)伴生对象概念
- 当单例对象的名称与某个类的名称一样时,该对象被称为这个类的伴生对象。类被称为该对象的伴生类。类和它的伴生对象必须定义在同一个文件中,且两者可以互相访问其私有成员。
(二)案例演示
- 在
net.huawei.day04
包里,创建Scala类Student
,在文件里创建其伴生对象Student
-
package net.zyf.day04
-
/**
-
* 功能:
-
* 作者:zyf
-
* 时间:2023年03月00号
-
*/
-
//伴生类
-
class Student {
-
private var name = "陈燕文"
-
def speak(): Unit = {
-
// 访问伴生对象的私有成员(Student.age)
-
println("我叫" + name + ",今年" + Student.age + "岁了~")
-
}
-
}
-
// 伴生对象
-
object Student {
-
private var age = 18
-
def main(args: Array[String]): Unit = {
-
// 基于伴生类创建学生对象
-
val student = new Student()
-
// 访问伴生类对象的私有成员(name)
-
println("姓名:" + student.name)
-
// 访问伴生对象的私有成员(age)
-
println("年龄:" + age) // 或者写成Student.age,但是不能写成student.age
-
// 调用伴生类对象的方法
-
student.speak()
-
}
-
}
四、get和set方法
(一)生成原则
- Scala默认会根据类的属性的修饰符生成不同的get和set方法
1、val修饰的属性 - 公共常量属性
- 系统会自动生成一个私有常量属性和一个公有get方法。
2、var修饰的属性 - 公共变量属性
- 系统会自动生成一个私有变量属性和一对公有get/set方法。
3、private var修饰的属性 - 私有变量属性
- 系统会自动生成一对私有get/set方法,相当于类的私有属性,只能在类的内部和伴生对象中使用。
4、private[this]修饰的属性 - 本地私有变量属性
- 系统不会生成get/set方法,即只能在类的内部使用该属性。
(二)案例演示
任务1、利用系统自动生成的get和set方法
- 在
net.huawei.day04
包里创建Dog
类
-
package net.zyf.day04
-
/**
-
* 功能:狗类
-
* 作者:zyf
-
* 日期:2023年03月日
-
*/
-
class Dog {
-
val id: Int = 1 // 系统会自动生成公共的get方法
-
var name: String = "瑞瑞" // 系统会自动生成公共的get和set方法
-
private var gender: String = "公" // 系统会自动生成私有的get和set方法,只有伴生对象可以访问
-
private[this] var age: Int = 5 // 系统不会生成get和set方法,即使伴生对象也无法访问
-
}
(2)编译成字节码文件
-
将
Dog.scala
编译成Dog.class
- 直接用记事本打开就是乱码
(3)将字节码文件反编译为Java代码
-
Java反编译工具 - Java Decompiler
-
package net.zyf.day04;
-
import scala.reflect.ScalaSignature;
-
@ScalaSignature(bytes = "\006\005}2Aa\003\007\001'!)!\004\001C\0017!9a\004\001b\001\n\003y\002BB\022\001A\003%\001\005C\004%\001\001\007I\021A\023\t\017E\002\001\031!C\001e!1\001\b\001Q!\n\031Bq!\017\001A\002\023%Q\005C\004;\001\001\007I\021B\036\t\ru\002\001\025)\003'\021\031q\004\001)Q\005A\t\031Ai\\4\013\0055q\021!\0023bsB\"$BA\b\021\003\rQ\030P\032\006\002#\005\031a.\032;\004\001M\021\001\001\006\t\003+ai\021A\006\006\002/\005)1oY1mC&\021\021D\006\002\007\003:L(+\0324\002\rqJg.\033;?)\005a\002CA\017\001\033\005a\021AA5e+\005\001\003CA\013\"\023\t\021cCA\002J]R\f1!\0333!\003\021q\027-\\3\026\003\031\002\"a\n\030\017\005!b\003CA\025\027\033\005Q#BA\026\023\003\031a$o\\8u}%\021QFF\001\007!J,G-\0324\n\005=\002$AB*ue&twM\003\002.-\005Aa.Y7f?\022*\027\017\006\0024mA\021Q\003N\005\003kY\021A!\0268ji\"9q'BA\001\002\0041\023a\001=%c\005)a.Y7fA\0051q-\0328eKJ\f!bZ3oI\026\024x\fJ3r)\t\031D\bC\0048\021\005\005\t\031\001\024\002\017\035,g\016Z3sA\005\031\021mZ3")
-
public class Dog {
-
private final int id = 1;
-
public int id() {
-
return this.id;
-
}
-
private String name = ";
-
public String name() {
-
return this.name;
-
}
-
public void name_$eq(String x$1) {
-
this.name = x$1;
-
}
-
private String gender = ";
-
private String gender() {
-
return this.gender;
-
}
-
private void gender_$eq(String x$1) {
-
this.gender = x$1;
-
}
-
private int age = 5;
-
}
(4)说明反编译生成的Java代码
- 使用name属性举例,在Scala中,get和set方法并非被命名为getName和setName,而是被命名为name和name_=,由于JVM不允许在方法名中出现=,因此=被翻译成$eq。
- 从上述代码可以看出,由于属性id使用val修饰,因此不可修改,只生成了与get方法对应的id();属性name使用var修饰,因此生成了与get和set方法对应的name()和name_$eq()方法,且都为public;属性gender由于使用private var修饰,因此生成了private修饰的get和set方法 - gender()和gender_$eq();属性age由于使用private[this]修饰,因此没有生成get和set方法,只能在类的内部使用。
(5)创建单例对象用来测试Dog类
- 创建
TestDog
单例对象
-
package net.zyf.day04
-
/**
-
* 功能:
-
* 作者:zyf
-
* 时间:2023年03月00号
-
*/
-
object TestDog {
-
def main(args: Array[String]): Unit = {
-
// 创建狗对象
-
val dog: Dog = new Dog()
-
// 访问id属性
-
println("id: " + dog.id)
-
// 设置name属性
-
dog.name = "欢欢"
-
// 访问name属性
-
println("name: " + dog.name)
-
}
-
}
- 注意:本地私有属性
age
,不能访问,即使伴生对象也无法访问
- 私有属性
gender
只有在伴生对象里可以访问,在非伴生对象TestDog
里是无法访问的
-
创建
Dog
类的伴生对象,可以访问属性id
和name
,就可以访问Dog
类的私有属性gender
-
package net.zyf.day04
-
/**
-
* 功能:狗类
-
* 作者:zyf
-
* 日期:2023年03月15日
-
*/
-
// 伴生类
-
class Dog {
-
val id: Int = 1 // 系统会自动生成公共的get方法
-
var name: String = "瑞瑞" // 系统会自动生成公共的get和set方法
-
private var gender: String = "公" // 系统会自动生成私有的get和set方法,只有伴生对象可以访问
-
private[this] var age: Int = 5 // 系统不会生成get和set方法,即使伴生对象也无法访问
-
}
-
// 伴生对象
-
object Dog {
-
def main(args: Array[String]): Unit = {
-
// 创建Dog对象
-
val dog: Dog = new Dog()
-
// 伴生对象访问id属性
-
println("id: " + dog.id)
-
// 伴生对象访问name属性
-
dog.name = "萌萌哒"
-
println("name: " + dog.name)
-
// 伴生对象设置伴生类对象的私有属性gender
-
dog.gender = "母"
-
// 伴生对象访问伴生类对象的私有属性gender
-
println("gender: " + dog.gender)
-
}
-
}
- 运行程序,查看结果
- 即使伴生对象也无法访问伴生类的本地私有属性
age
- 类似的效果,用Java来实现
任务2、用户自己编写私有属性的Scala风格的get和set方法
- 注意:set方法的写法 ——
方法名_=
- 在
net.huawei.day04
包里创建Cat
类
-
package net.zyf.day04
-
/**
-
* 功能:猫类
-
* 作者:zyf
-
* 日期:2023年03月24日
-
*/
-
// 伴生类
-
class Cat {
-
private var catName: String = "虎丸"
-
// 定义get方法 - name
-
def name: String = catName // 相当于Java里的getCatName方法
-
// 定义set方法 - name_=
-
def name_=(name: String): Unit = { // 相当于Java里的setCatName方法
-
catName = name
-
}
-
}
-
// 伴生对象
-
object Cat {
-
def main(args: Array[String]): Unit = {
-
// 创建猫对象
-
val cat: Cat = new Cat()
-
// 获取对象属性
-
println("猫原来的名字:" + cat.name)
-
// 设置对象属性
-
cat.name_=("冰轮丸")
-
// 访问伴生类对象私有属性
-
println("猫现在的名字:" + cat.catName)
-
}
-
}
- 运行结果
任务3、用户自己编写私有属性的Java风格的get和set方法
- 注意:get方法 —— getXXX(),set方法——setXXX()
- 在
net.huawei.day04
包里创建Bird
类
package net.zyf.day04
/**
* 功能:
* 作者:zyf
* 时间:2023年03月00号
*/
object Bird {
// 伴生类
class Bird {
private var name = "玲玲"
// Java风格的get方法
def getName: String = name
// Java风格的set方法
def setName(name: String): Unit = {
this.name = name
}
}
// 伴生对象
object Bird {
def main(args: Array[String]): Unit = {
// 创建鸟对象
val bird = new Bird()
// 获取对象属性
println("鸟原来的名字:" + bird.getName)
// 设置对象属性
bird.setName("菲菲")
// 伴生对象访问伴生类对象私有属性
println("鸟现在的名字:" + bird.name)
}
}
}
五、构造器
- Scala中的构造器分为主构造器和辅助构造器。
(一)主构造器
1、构造器参数带val或var
- 主构造器的参数直接放在类名之后,且将被编译为类的成员变量,其值在初始化类时传入。注意,构造器参数必须指定类型
- 在
net.huawei.day05
包里创建Person
类
package net.zyf.day05
/**
* 功能:
* 作者:zyf
* 时间:2023年03月00号
*/
// 伴生类
class Person (val name: String, var age: Int = 18) {
def speak(): Unit = println("我叫" + name + ",今年" + age + "岁了~")
}
// 伴生对象
object Person {
def main(args: Array[String]): Unit = {
// 基于主构造器创建对象,传入两个参数值
val person = new Person("李文华", 25)
// 访问构造器参数,其实是访问成员属性
println("姓名:" + person.name)
println("年龄:" + person.age)
// 调用对象的方法
person.speak()
// 修改对象属性(name属性不能改,age属性可以改)
person.age_=(26)
// 再次调用对象的方法
person.speak()
}
}
- 修改对象的
name
属性,要报错,因为主构造器的name
参数是常量,所以类不提供设置属性的方法
- 因为主构造器第二个参数是有初始值,所以创建对象时可以不再初始化第二个参数,采用主构造器参数的初始值
2、构造器参数带访问权限
- 可以通过对主构造器的参数添加访问修饰符来控制参数的访问权限
- 创建Person类,将参数age设置为私有的,参数name设置为不可修改(val)
-
class Person (val name: String, private var age: Int) {
-
}
- 系统会为
name
属性添加一个公共的get方法 - 系统会为
age
属性添加一个私有的get和set方法
3、构造器参数不带var或val
- 构造参数也可以不带
val
或var
,此时默认为private[this] val
,这样会被编译成类的本地私有成员,不会生成get
和set
方法,只能在类的内部访问。
-
class Person (name: String, age: Int) {
-
}
- 可以改写成带访问权限的参数
-
class Person (private[this] val name: String, private[this] val age: Int) {
-
}
4、类的初始化语句
- 主构造器执行时,类中定义的语句作为初始化语句
- 在
net.huawei.day05
包里创建Dog
类
-
package net.zyf.day05
-
/**
-
* 功能:
-
* 作者:zyf
-
* 时间:2023年03月00号
-
*/
-
class Dog (var name: String, var age: Int){
-
name = "瑞瑞"
-
age = 5
-
println("主构造器被调用了~")
-
def speak(): Unit = println("我叫" + name + ",今年" + age + "岁了~")
-
}
-
// 伴生对象
-
object Dog {
-
def main(args: Array[String]): Unit = {
-
val dog = new Dog("", 0)
-
dog.speak()
-
}
-
}
- 说明:实例化
Dog
时,传入的参数是""
与0
,但是会执行类里的两个给成员变量赋值的语句,于是name
成了瑞瑞
,age
成了5
,于是调用对象的speak()
方法,会输出我叫瑞瑞,今年5岁了~
。
5、私有化构造器
- 如果需要将整个主构造器设置为私有的,那么只需要添加
private
关键字即可,注意,只有伴生对象里才能调用私有构造器来实例化,非伴生对象里就不能调用私有构造器来实例化 - 在
net.huawei.day05
包里创建Cat
类
-
package net.zyf.day05
-
/**
-
* 功能:类
-
* 作者:zyf
-
* 日期:2023年03月日
-
*/
-
// 伴生类
-
class Cat private (var name: String, var age: Int) {
-
def speak(): Unit = {
-
println("我叫" + name + ",今年" + age + "岁了~")
-
}
-
}
-
// 伴生对象
-
object Cat {
-
def main(args: Array[String]): Unit = {
-
val cat = new Cat("欢欢", 2)
-
cat.speak()
-
}
-
}
- 如果改成非伴生对象
TestCat
,报错:私有构造器不能被非伴生对象TestCat
访问
6、无参构造器
- 主构造器也可以没有参数,一个类中如果没有显式地定义主构造器,就默认有一个无参构造器。
- 在
net.huawei.day05
包里创建Bird
类
-
package net.zyf.day05
-
/**
-
* 功能:鸟类
-
* 作者:zyf
-
* 日期:2023年03月日
-
*/
-
// 伴生类
-
class Bird () { // 显式定义无参构造器
-
var name = "玲玲"
-
var age = 4
-
def speak(): Unit = {
-
println("我叫" + name + ",今年" + age + "岁了~")
-
}
-
}
-
// 伴生对象
-
object Bird {
-
def main(args: Array[String]): Unit = {
-
val bird = new Bird() // 调用无参构造器实例化
-
bird.speak()
-
}
-
}
(二)辅助构造器
- Scala类除了可以有主构造器外,还可以有任意多个辅助构造器。
1、定义辅助构造器的注意事项
- 辅助构造器的方法名称为
this
- 每一个辅助构造器的方法体中必须首先调用其他已定义的构造器
- 辅助构造器的参数不能使用
var
或val
进行修饰
2、案例演示
- 在
net.hw.constructor
包里创建Student
类
-
package net.zyf.day05
-
/**
-
* 功能:演示辅助构造器
-
* 作者:zyf
-
* 时间:2023年03月00号
-
*/
-
// 伴生类
-
class Student {
-
private var name = "李林芮"
-
private var age = 18
-
// 定义单参辅助构造器
-
def this(name: String) = {
-
this() // 调用无参主构造器
-
this.name = name
-
}
-
// 定义双参辅助构造器
-
def this(name: String, age: Int) = {
-
this(name) // 调用单参辅助构造器
-
this.age = age
-
}
-
// 重写toString方法
-
override def toString: String = "我叫" + name + ",今年" + age + "岁了~"
-
}
-
// 伴生对象
-
object Student {
-
def main(args: Array[String]): Unit = {
-
// 调用无参构造器实例化
-
val student1 = new Student()
-
println(student1)
-
// 调用单参辅助构造器实例化
-
val student2 = new Student("王晓琳")
-
println(student2)
-
// 调用双参辅助构造器实例化
-
val student3 = new Student("张智霖", 21)
-
println(student3)
-
}
-
}
运行结果
- 课堂练习:用Java程序来完成上述任务
(2)有参主构造器与有参辅助构造器
- 主构造器还可以与辅助构造器同时使用,在这种情况下,一般辅助构造器的参数要多于主构造器
- 在
net.huawei.day05
包里创建Teacher
类
-
package net.zyf.day05
-
/**
-
* 功能:教师类
-
* 作者:zyf
-
* 时间:2023年03月00号
-
*/
-
class Teacher (private var name: String, private var age: Int){ // 双参主构造器
-
private var gender = ""
-
// 三参辅助构造器
-
def this(name: String, age: Int, gender: String) = {
-
this(name, age) // 调用双参主构造器
-
this.gender = gender
-
}
-
// 重写toString方法
-
override def toString: String = "我叫" + name + "," + gender + ",今年" + age + "岁了~"
-
}
-
// 伴生对象
-
object Teacher {
-
def main(args: Array[String]): Unit = {
-
// 调用三参辅助构造器实例化
-
val teacher = new Teacher("无心剑", 50, "男")
-
println(teacher)
-
}
-
}
六、抽象类
(一)抽象类的定义
- Scala的抽象类使用关键字
abstract
定义
-
abstract class 类名 {
-
}
(二)抽象类的特征
- 抽象类不能被实例化。
- 抽象类中可以定义抽象字段(没有初始化的字段)和抽象方法(没有被实现的方法),也可以定义被初始化的字段和被实现的方法。
- 若某个子类继承了一个抽象类,则必须实现抽象类中的抽象字段和抽象方法,且实现的过程中可以添加override关键字,也可以省略。若重写了抽象类中已经实现的方法,则必须添加override关键字。
(三)案例演示
1、创建抽象类 - Person
- 在
net.huawei.day06
包里创建Person
抽象类
-
package net.zyf.day06
-
/**
-
* 功能:抽象人类类
-
* 作者:zyf
-
* 日期:2023年03月日
-
*/
-
abstract class Person {
-
var name: String // 抽象字段
-
var age: Int // 抽象字段
-
var address: String = "龙马潭区长桥路2号" // 普通字段
-
// 抽象方法
-
def speak()
-
// 普通方法
-
def walk(): Unit = {
-
println(name + "在散步~")
-
}
-
}
2、继承抽象类,创建普通类 - Teacher
- 在
net.huawei.day06
包里创建Teacher
普通类
3、创建测试对象 - TestTeacher
- 在
net.huawei.day06
包里创建TestTeacher
对象
4、运行程序,查看结果
- 运行
TestTeacher
对象
5、简要说明
- 需要注意的是,上述Teacher类中speak()方法的地址字段(address)是从父类(抽象类Person)中继承而来的。由于该字段在Person中有初始化值,不是抽象字段,若需要在Teacher类中修改该字段的值,则可以在Teacher类的构造函数或其它方法中使用this.address对其重新赋值。例如,将地址改为“江阳区前进中路3号”,可以使用以下代码:this.address="江阳区前进中路3号"。
七、特质
(一)特质的概念
- Scala特质使用关键字trait定义,类似Java 8中使用interface定义的接口。特质除了有Java接口的功能外,还有一些特殊的功能。Scala特质中,字段和方法的定义与Scala抽象类一样,可以定义抽象字段和抽象方法、非抽象字段和非抽象方法。
(二)特质的定义
1、语法格式
-
trait 特质名 {
-
// 抽象字段
-
// 抽象方法
-
// 普通字段
-
// 普通方法
-
}
2、案例演示
- 在
net.huawei.day07
包里创建Pet
特质
任务2、创建奔跑特质 - Runnable
- 在
net.huawei.day07
包里创建Runnable
特质
任务3、创建飞翔特质 - Flyable
- 在
net.huawei.day07
包里创建Flyable
特质
(三)特质的实现
- 类可以使用关键字
extends
实现特质,但必须实现特质中未实现的字段和方法(抽象字段和抽象方法),这一点与继承抽象类是一致的。
1、语法格式
-
class 类名 extends 特质名 {
-
// 实现抽象字段
-
// 实现抽象方法
-
}
(2)实现多个特质
- 如果需要实现的特质不止一个,那么可以通过
with
关键字添加额外特质,但位于最左侧的特质必须使用extends
关键字。
-
class 类名 extends 特质名1 with 特质名2 with 特质名3 …… with 特质名n {
-
// 实现抽象字段
-
// 实现抽象方法
-
}
2、案例演示
- 在
net.huawei.day07
包里创建Cat
类,实现Pet
特质
- 运行程序,查看结果
任务2、实现多个特质
-
在
net.huawei.day07
包里创建Bird
类,实现Pet
、Runnable
、Flyable
特质
- 运行程序,查看结果