当前位置: 首页 > news >正文

Scala中类的继承、抽象类和特质

1. 类的继承

1.1 Scala中的继承结构

Scala 中继承关系如下图:

+ Any 是整个继承关系的根节点;

+ AnyRef 包含 Scala Classes 和 Java Classes,等价于 Java 中的 java.lang.Object;

+ AnyVal 是所有值类型的一个标记;

+ Null 是所有引用类型的子类型,唯一实例是 null,可以将 null 赋值给除了值类型外的所有类型的变量;

+ Nothing 是所有类型的子类型。

1.2 extends & override

Scala 的集成机制和 Java 有很多相似之处,比如都使用 `extends` 关键字表示继承,都使用 `override` 关键字表示重写父类的方法或成员变量。示例如下:

//父类
class Person {var name = ""// 1.不加任何修饰词,默认为 public,能被子类和外部访问var age = 0// 2.使用 protected 修饰的变量能子类访问,但是不能被外部访问protected var birthday = ""// 3.使用 private 修饰的变量不能被子类和外部访问private var sex = ""  def setSex(sex: String): Unit = {this.sex = sex}// 4.重写父类的方法建议使用 override 关键字修饰override def toString: String = name + ":" + age + ":" + birthday + ":" + sex
}

使用 `extends` 关键字实现继承:

// 1.使用 extends 关键字实现继承
class Employee extends Person {override def toString: String = "Employee~" + super.toString// 2.使用 public 或 protected 关键字修饰的变量能被子类访问
  def setBirthday(date: String): Unit = {
    birthday = date}
}

测试继承:


object ScalaApp extends App {val employee = new Employee  employee.name = "shangjack"
  employee.age = 20
  employee.setBirthday("2019-03-05")
  employee.setSex("男")println(employee)
}// 输出: Employee~shangjack:20:2019-03-05:男

1.3 调用超类构造器

在 Scala 的类中,每个辅助构造器都必须首先调用其他构造器或主构造器,这样就导致了子类的辅助构造器永远无法直接调用超类的构造器,只有主构造器才能调用超类的构造器。所以想要调用超类的构造器,代码示例如下:

class Employee(name:String,age:Int,salary:Double) extends Person(name:String,age:Int) {.....
}

1.4 类型检查和转换

想要实现类检查可以使用 `isInstanceOf`,判断一个实例是否来源于某个类或者其子类,如果是,则可以使用 `asInstanceOf` 进行强制类型转换。

object ScalaApp extends App {val employee = new Employeeval person = new Person// 1. 判断一个实例是否来源于某个类或者其子类 输出 true println(employee.isInstanceOf[Person])println(person.isInstanceOf[Person])// 2. 强制类型转换var p: Person = employee.asInstanceOf[Person]// 3. 判断一个实例是否来源于某个类 (而不是其子类)println(employee.getClass == classOf[Employee])
}

1.5 构造顺序和提前定义

1.5.1 构造顺序

在 Scala 中还有一个需要注意的问题,如果你在子类中重写父类的 val 变量,并且超类的构造器中使用了该变量,那么可能会产生不可预期的错误。下面给出一个示例:

// 父类
class Person {println("父类的默认构造器")val range: Int = 10val array: Array[Int] = new Array[Int](range)
}//子类
class Employee extends Person {println("子类的默认构造器")override val range = 2
}//测试
object ScalaApp extends App {val employee = new Employeeprintln(employee.array.mkString("(", ",", ")"))
}

这里初始化 array 用到了变量 range,这里你会发现实际上 array 既不会被初始化 Array(10),也不会被初始化为 Array(2),实际的输出应该如下:

父类的默认构造器
子类的默认构造器
()

可以看到 array 被初始化为 Array(0),主要原因在于父类构造器的执行顺序先于子类构造器,这里给出实际的执行步骤:

1. 父类的构造器被调用,执行 `new Array[Int](range)` 语句;

2. 这里想要得到 range 的值,会去调用子类 range() 方法,因为 `override val` 重写变量的同时也重写了其 get 方法;

3. 调用子类的 range() 方法,自然也是返回子类的 range 值,但是由于子类的构造器还没有执行,这也就意味着对 range 赋值的 `range = 2` 语句还没有被执行,所以自然返回 range 的默认值,也就是 0。

这里可能比较疑惑的是为什么 `val range = 2` 没有被执行,却能使用 range 变量,这里因为在虚拟机层面,是先对成员变量先分配存储空间并赋给默认值,之后才赋予给定的值。想要证明这一点其实也比较简单,代码如下:

class Person {// val range: Int = 10 正常代码 array 为 Array(10)val array: Array[Int] = new Array[Int](range)val range: Int = 10  //如果把变量的声明放在使用之后,此时数据 array 为 array(0)
}object Person {
  def main(args: Array[String]): Unit = {val person = new Personprintln(person.array.mkString("(", ",", ")"))}
}

1.5.2 提前定义

想要解决上面的问题,有以下几种方法:

(1) . 将变量用 final 修饰,代表不允许被子类重写,即 `final val range: Int = 10 `;

(2) . 将变量使用 lazy 修饰,代表懒加载,即只有当你实际使用到 array 时候,才去进行初始化;

lazy val array: Array[Int] = new Array[Int](range)

(3) . 采用提前定义,代码如下,代表 range 的定义优先于超类构造器。

class Employee extends {//这里不能定义其他方法override val range = 2
} with Person {// 定义其他变量或者方法
  def pr(): Unit = {println("Employee")}
}

但是这种语法也有其限制:你只能在上面代码块中重写已有的变量,而不能定义新的变量和方法,定义新的变量和方法只能写在下面代码块中。

类的继承和下文特质 (trait) 的继承都存在这个问题,也同样可以通过提前定义来解决。虽然如此,但还是建议合理设计以规避该类问题。

2. 抽象类

Scala 中允许使用 `abstract` 定义抽象类,并且通过 `extends` 关键字继承它。

定义抽象类:

abstract class Person {// 1.定义字段var name: Stringval age: Int// 2.定义抽象方法
  def geDetail: String// 3. scala 的抽象类允许定义具体方法
  def print(): Unit = {println("抽象类中的默认方法")}
}

继承抽象类:

class Employee extends Person {// 覆盖抽象类中变量override var name: String = "employee"override val age: Int = 12// 覆盖抽象方法
  def geDetail: String = name + ":" + age
}

3. 特质

3.1 trait & with

Scala 中没有 interface 这个关键字,想要实现类似的功能,可以使用特质 (trait)。trait 等价于 Java 8 中的接口,因为 trait 中既能定义抽象方法,也能定义具体方法,这和 Java 8 中的接口是类似的。

// 1.特质使用 trait 关键字修饰
trait Logger {// 2.定义抽象方法
  def log(msg: String)// 3.定义具体方法
  def logInfo(msg: String): Unit = {println("INFO:" + msg)}
}

想要使用特质,需要使用 `extends` 关键字,而不是 `implements` 关键字,如果想要添加多个特质,可以使用 `with` 关键字。

// 1.使用 extends 关键字,而不是 implements,如果想要添加多个特质,可以使用 with 关键字
class ConsoleLogger extends Logger with Serializable with Cloneable {// 2. 实现特质中的抽象方法
  def log(msg: String): Unit = {println("CONSOLE:" + msg)}
}

3.2 特质中的字段

和方法一样,特质中的字段可以是抽象的,也可以是具体的:

如果是抽象字段,则混入特质的类需要重写覆盖该字段;

如果是具体字段,则混入特质的类获得该字段,但是并非是通过继承关系得到,而是在编译时候,简单将该字段加入到子类。

trait Logger {// 抽象字段var LogLevel:String// 具体字段var LogType = "FILE"
}/覆盖抽象字段:
class InfoLogger extends Logger {// 覆盖抽象字段override var LogLevel: String = "INFO"
}

3.3 带有特质的对象

Scala 支持在类定义的时混入 ` 父类 trait`,而在类实例化为具体对象的时候指明其实际使用的 ` 子类 trait`。示例如下:

trait Logger:
// 父类
trait Logger {// 定义空方法 日志打印
  def log(msg: String) {}
}// trait ErrorLogger:// 错误日志打印,继承自 Logger
trait ErrorLogger extends Logger {// 覆盖空方法override def log(msg: String): Unit = {println("Error:" + msg)}
}/trait InfoLogger:// 通知日志打印,继承自 Logger
trait InfoLogger extends Logger {// 覆盖空方法override def log(msg: String): Unit = {println("INFO:" + msg)}
}

具体的使用类:

// 混入 trait Logger
class Person extends Logger {// 调用定义的抽象方法
  def printDetail(detail: String): Unit = {log(detail)}
}

这里通过 main 方法来测试:

object ScalaApp extends App {// 使用 with 指明需要具体使用的 trait  val person01 = new Person with InfoLoggerval person02 = new Person with ErrorLoggerval person03 = new  Person with InfoLogger with ErrorLogger
  person01.log("scala")  //输出 INFO:scala
  person02.log("scala")  //输出 Error:scala
  person03.log("scala")  //输出 Error:scala}

这里前面两个输出比较明显,因为只指明了一个具体的 `trait`,这里需要说明的是第三个输出,因为 trait 的调用是由右到左开始生效的**,所以这里打印出 `Error:scala`。

3.4 特质构造顺序

`trait` 有默认的无参构造器,但是不支持有参构造器。一个类混入多个特质后初始化顺序应该如下:

// 示例
class Employee extends Person with InfoLogger with ErrorLogger {...}

1. 超类首先被构造,即 Person 的构造器首先被执行;

2. 特质的构造器在超类构造器之前,在类构造器之后;特质由左到右被构造;每个特质中,父特质首先被构造;

Logger 构造器执行(Logger 是 InfoLogger 的父类);

InfoLogger 构造器执行;

ErrorLogger 构造器执行;

3. 所有超类和特质构造完毕,子类才会被构造。

相关文章:

Scala中类的继承、抽象类和特质

1. 类的继承 1.1 Scala中的继承结构 Scala 中继承关系如下图: Any 是整个继承关系的根节点; AnyRef 包含 Scala Classes 和 Java Classes,等价于 Java 中的 java.lang.Object; AnyVal 是所有值类型的一个标记; Nul…...

小程序如何实现登录数据持久化

在小程序中实现登录数据的持久化可以通过以下几种方式: 使用本地缓存 在用户登录成功后,将登录凭证或用户信息等数据使用 wx.setStorageSync 方法存储到本地缓存中: // 存储登录数据到本地缓存 wx.setStorageSync(token, 登录凭证); wx.set…...

Maven本地配置获取nexus私服的依赖

场景 Nexus-在项目中使用Maven私服,Deploy到私服、上传第三方jar包、在项目中使用私服jar包: Nexus-在项目中使用Maven私服,Deploy到私服、上传第三方jar包、在项目中使用私服jar包_nexus maven-releases 允许deploy-CSDN博客 在上面讲的是…...

第02章-变量与运算符

1 关键字 关键字:被Java语言赋予了特殊含义,用作专门用途的字符串(或单词)。如class、public、static、void等,这些单词都被Java定义好了,称为关键字。 特点:关键字都是小写字母;官…...

SpringBoot数据响应、分层解耦、三层架构

响应数据 ResponseBody 类型:方法注解、类注解位置:Controller方法、类上作用:将方法返回值直接响应,如果返回值类型是 实体对象/集合 ,将会转换为json格式响应说明:RestController Controller Respons…...

go测试库之apitest

📢专注于分享软件测试干货内容,欢迎点赞 👍 收藏 ⭐留言 📝 如有错误敬请指正!📢交流讨论:欢迎加入我们一起学习!📢资源分享:耗时200小时精选的「软件测试」资…...

K8S删除资源后一直处于Terminating状态无法删除解决方法

原因 使用kubectl delete 删除某命名空间是一直处于Terminating状态无法删除,首先排查了该命名空间下是否还存在deployment pod等资源发现没有后,等了很久还是无法删除后发现是因为该名称空间的“finalizers”字段有值导致 Finalizer(终结器…...

jvm实践

说一下JVM中的分代回收 堆的区域划分 1.堆被分为了两份:新生代和老年代[1:2] 2.对于新生代,内部又被分为了三个区域。Eden区,幸存者区survivor(分成from和to)[8:1:1] 对象回收分代回收策略 1.新创建的对象,都会先分配到eden区 2.当伊园内存…...

redis-plus-plus访问REDIS集群

编程语言:C 开源库:redis-plus-plus 接口类:RedisCluster 初始化需要输入任意一个结点的IP和端口,如果设置了密码,还需要密码的明文并使用ConnectionOptions类。 初始化完成后可以直接进行读/写操作。 RedisClust…...

python把Word题库转成Excle题库

又到了一年一度的背题时刻,但是收到的题库是Word版的,页数特别多 话不多说,上代码,有图有真相,代码里面备注的很详细 # 导入所需库 import csv import os import refrom docx import Document from win32com import c…...

算法通关村第六关-白银挑战树

大家好我是苏麟 , 今天聊聊树 . 大纲 树的概念二叉树满二叉树完全二叉树 树的性质树的定义与存储方式树的遍历通过序列构造二叉树前中序列遍历 树的概念 树是我们计算机中非常重要的一种数据结构,同时使用树这种数据结构,可以描述现实生活中的很多事物&…...

【Java对象】一文读懂 Java 对象庐山真面目及指针压缩

文章目录 版本及工具介绍Java 对象结构对象头mark word 标记字mark word 标记字解析Lock Record class point 类元数据指针 实例数据对齐填充为什么需要对齐填充 常见 Java 数据类型对象分析ArrayListLongStringByteBoolean 其它指针压缩前置知识:32位操作系统为什么…...

leetcode做题笔记210. 课程表 II

现在你总共有 numCourses 门课需要选,记为 0 到 numCourses - 1。给你一个数组 prerequisites ,其中 prerequisites[i] [ai, bi] ,表示在选修课程 ai 前 必须 先选修 bi 。 例如,想要学习课程 0 ,你需要先完成课程 1…...

【深度学习 AIGC】stable diffusion webUI 使用过程,参数设置,教程,使用方法

文章目录 docker快速启动vae.ckpt或者.safetensorsCFG指数/CFG Scale面部修复/Restore facesRefinerTiled VAEClip Skipprompt提示词怎么写 docker快速启动 如果你想使用docker快速启动这个项目,你可以按下面这么操作(显卡支持CUDA11.8)。如…...

论文阅读 - Detecting Social Bot on the Fly using Contrastive Learning

目录 摘要: 引言 3 问题定义 4 CBD 4.1 框架概述 4.2 Model Learning 4.2.1 通过 GCL 进行模型预训练 4.2.2 通过一致性损失进行模型微调 4.3 在线检测 5 实验 5.1 实验设置 5.2 性能比较 5.5 少量检测研究 6 结论 https://dl.acm.org/doi/pdf/10.1145/358…...

PaddleMIX学习笔记(1)

写在前面 之前对HyperLedger的阅读没有完全结束,和很多朋友一样,同时也因为工作的需要,最近开始转向LLM方向。 国内在大模型方面生态做的最好的,目前还是百度的PaddlePaddle,所以自己也就先从PP开始看起了。 众所周知…...

【网络协议】聊聊HTTPS协议

前面的文章,我们描述了网络是怎样进行传输数据包的,但是网络是不安全的,对于这种流量门户网站其实还好,对于支付类场景其实容易将数据泄漏,所以安全的方式是通过加密,加密方式主要是对称加密和非对称加密。…...

2023.11.2事件纪念

然而造化又常常为庸人设计,以时间的流逝,来洗涤旧迹,仅以留下淡红的血色和微漠的悲哀。 回顾这次事件,最深的感触就是什么是团队的力量! 当我们看到希望快要成功的时候,大家洋溢出兴奋开心的表情,一起的欢声笑语;但看…...

Scala和Play WS库编写的爬虫程序

使用Scala和Play WS库编写的爬虫程序,该程序将爬取网页内容: import play.api.libs.ws._ import scala.concurrent.ExecutionContext.Implicits.global ​ object BaiduCrawler {def main(args: Array[String]): Unit {val url ""val proxy…...

佳易王配件进出库开单打印进销存管理系统软件下载

用版配件进出库开单打印系统,可以有效的管理:供货商信息,客户信息,进货入库打印,销售出库打印,进货明细或汇总统计查询,销售出库明细或汇总统计查询,库存查询,客户往来账…...

【深度学习基础】专业术语汇总(欠拟合和过拟合、泛化能力与迁移学习、调参和超参数、训练集、测试集和验证集)

📢:如果你也对机器人、人工智能感兴趣,看来我们志同道合✨ 📢:不妨浏览一下我的博客主页【https://blog.csdn.net/weixin_51244852】 📢:文章若有幸对你有帮助,可点赞 👍…...

【C语言:函数栈帧的创建与销毁】

文章目录 前言一、前期准备1.寄存器2.汇编指令3.测试代码 二、解开函数栈帧的神秘面纱1.栈帧大体轮廓2.main函数栈帧的创建3.main函数内执行有效代码4.烫烫烫5.函数参数的传递6.add函数栈帧的创建7.add函数内执行有效代码8.add是如何获得参数的9. add函数栈帧的销毁10.main函数…...

怎么在C++中实现云端存储变量

随着云计算技术的快速发展,现在我们可以将数据存储在云端,以便于在不同设备和地点访问。在C中,我们也可以通过一些方法来实现这个功能。本文将详细介绍如何在C中实现云端存储变量。 首先,我们需要理解,C本身并没有直接…...

短视频矩阵营销系统工具如何助力商家企业获客?

1.批量剪辑技术研发 做的数学建模算法,数学阶乘的组合乘组形式,采用两套查重机制,一套针对素材进行查重抽帧素材,一套针对成片进行抽帧素材打分制度查重,自动滤重计入打分。 2.账号矩阵分发开发 多平台,…...

PCL 计算一个平面与包围盒体素的相交线

文章目录 一、简介二、实现代码三、实现效果参考资料一、简介 基于之前计算的包围盒体素(PCL 包围盒体素化显示),这里使用一个平面与其进行相交,并求出与其中体素单元的相交线。 二、实现代码 //标准文件 #include <iostream> #include <thread>//PCL...

面向教育的计算机视觉和深度学习5

面向教育的计算机视觉和深度学习5 1. 好处智能内容&#xff08;Smart Content&#xff09;任务自动化&#xff08;Task Automation&#xff09;缩小技能差距&#xff08;Closing Skill Gap&#xff09; 2. 应用程序学生学习与福利&#xff08;Student Learning and Welfare&…...

FPGA芯片内部结构

参考链接&#xff1a;FPGA的进阶之第二章FPGA芯片内部结构&#xff08;2&#xff09;...

人工智能AI创作系统ChatGPT网站系统源码+AI绘画系统支持GPT4.0/支持Midjourney局部重绘

一、前言 SparkAi创作系统是基于OpenAI很火的ChatGPT进行开发的Ai智能问答系统和Midjourney绘画系统&#xff0c;支持OpenAI-GPT全模型国内AI全模型。本期针对源码系统整体测试下来非常完美&#xff0c;可以说SparkAi是目前国内一款的ChatGPT对接OpenAI软件系统。那么如何搭建…...

Google 开源项目风格指南

目录 C 风格指南 Objective-C 风格指南 Python 风格指南 Shell 风格指南 TypeScript 风格指南 Javascript 风格指南 HTML/CSS 风格指南 C 风格指南 C 风格指南 - 内容目录 — Google 开源项目风格指南 Objective-C 风格指南 Objective-C 风格指南 - 内容目录 — Googl…...

无限上下文,多级内存管理!突破ChatGPT等大语言模型上下文限制

目前&#xff0c;ChatGPT、Llama 2、文心一言等主流大语言模型&#xff0c;因技术架构的问题上下文输入一直受到限制&#xff0c;即便是Claude 最多只支持10万token输入&#xff0c;这对于解读上百页报告、书籍、论文来说非常不方便。 为了解决这一难题&#xff0c;加州伯克利…...