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

Scala的类和对象

1. 初识类和对象

Scala 的类与 Java 的类具有非常多的相似性,示例如下:

// 1. 在 scala 中,类不需要用 public 声明,所有的类都具有公共的可见性
class Person {// 2. 声明私有变量,用 var 修饰的变量默认拥有 getter/setter 属性private var age = 0// 3.如果声明的变量不需要进行初始赋值,此时 Scala 就无法进行类型推断,所以需要显式指明类型private var name: String = _// 4. 定义方法,应指明传参类型。返回值类型不是必须的,Scala 可以自动推断出来,但是为了方便调用者,建议指明def growUp(step: Int): Unit = {
    age += step}// 5.对于改值器方法 (即改变对象状态的方法),即使不需要传入参数,也建议在声明中包含 ()def growUpFix(): Unit = {
    age += 10}// 6.对于取值器方法 (即不会改变对象状态的方法),不必在声明中包含 ()def currentAge: Int = {
    age}/**
   * 7.不建议使用 return 关键字,默认方法中最后一行代码的计算结果为返回值
   *   如果方法很简短,甚至可以写在同一行中
   */def getName: String = name}// 伴生对象
object Person {def main(args: Array[String]): Unit = {// 8.创建类的实例val counter = new Person()// 9.用 var 修饰的变量默认拥有 getter/setter 属性,可以直接对其进行赋值
    counter.age = 12
    counter.growUp(8)
    counter.growUpFix()// 10.用 var 修饰的变量默认拥有 getter/setter 属性,可以直接对其进行取值,输出: 30
    println(counter.age)// 输出: 30
    println(counter.currentAge)// 输出: null
    println(counter.getName)}
}

2. 类

2.1 成员变量可见性

Scala 中成员变量的可见性默认都是 public,如果想要保证其不被外部干扰,可以声明为 private,并通过 getter 和 setter 方法进行访问。

2.2 getter和setter属性

getter 和 setter 属性与声明变量时使用的关键字有关:

使用 var 关键字:变量同时拥有 getter 和 setter 属性;

使用 val 关键字:变量只拥有 getter 属性;

使用 private[this]:变量既没有 getter 属性、也没有 setter 属性,只能通过内部的方法访问;

需要特别说明的是:假设变量名为 age,则其对应的 get 和 set 的方法名分别叫做 ` age` 和 `age_=`。

class Person {private val name = "heibaiying"private var age = 12private[this] var birthday = "2019-08-08"// birthday 只能被内部方法所访问def getBirthday: String = birthday
}object Person {def main(args: Array[String]): Unit = {val person = new Person
    person.age = 30
    println(person.name)
    println(person.age)
    println(person.getBirthday)}
}

示例代码中 `person.age=30` 在执行时内部实际是调用了方法 `person.age_=(30) `,而 `person.age` 内部执行时实际是调用了 `person.age()` 方法。想要证明这一点,可以对代码进行反编译。同时为了说明成员变量可见性的问题,我们对下面这段代码进行反编译:

class Person {
var name = ""
private var age = ""
}

依次执行下面编译命令:

scalac Person.scala
javap -private Person

编译结果如下,从编译结果可以看到实际的 get 和 set 的方法名 (因为 JVM 不允许在方法名中出现=,所以它被翻译成$eq),同时也验证了成员变量默认的可见性为 public。

Compiled from "Person.scala"
public class Person {
private java.lang.String name;
private java.lang.String age;public java.lang.String name();
public void name_$eq(java.lang.String);private java.lang.String age();
private void age_$eq(java.lang.String);public Person();
}

2.3 @BeanProperty

在上面的例子中可以看到我们是使用 `.` 来对成员变量进行访问的,如果想要额外生成和 Java 中一样的 getXXX 和 setXXX 方法,则需要使用@BeanProperty 进行注解。

class Person {@BeanProperty var name = ""
}object Person {def main(args: Array[String]): Unit = {val person = new Person
    person.setName("heibaiying")
    println(person.getName)}
}

2.4 主构造器

和 Java 不同的是,Scala 类的主构造器直接写在类名后面,但注意以下两点

主构造器传入的参数默认就是 val 类型的,即不可变,你没有办法在内部改变传参;

写在主构造器中的代码块会在类初始化的时候被执行,功能类似于 Java 的静态代码块 `static{}`

class Person(val name: String, val age: Int) {  println("功能类似于 Java 的静态代码块 static{}")def getDetail: String = {//name="heibai" 无法通过编译
    name + ":" + age}
}object Person {def main(args: Array[String]): Unit = {val person = new Person("heibaiying", 20)
    println(person.getDetail)}
}输出:
功能类似于 Java 的静态代码块 static{}
heibaiying:20

2.5 辅助构造器

辅助构造器有两点硬性要求:

辅助构造器的名称必须为 this;

每个辅助构造器必须以主构造器或其他的辅助构造器的调用开始。

class Person(val name: String, val age: Int) {private var birthday = ""// 1.辅助构造器的名称必须为 thisdef this(name: String, age: Int, birthday: String) {// 2.每个辅助构造器必须以主构造器或其他的辅助构造器的调用开始this(name, age)this.birthday = birthday}// 3.重写 toString 方法override def toString: String = name + ":" + age + ":" + birthday
}object Person {def main(args: Array[String]): Unit = {
    println(new Person("heibaiying", 20, "2019-02-21"))}
}

2.6 方法传参不可变

在 Scala 中,方法传参默认是 val 类型,即不可变,这意味着你在方法体内部不能改变传入的参数。这和 Scala 的设计理念有关,Scala 遵循函数式编程理念,强调方法不应该有副作用。

class Person() {def low(word: String): String = {
    word="word" // 编译无法通过
    word.toLowerCase}
}

3. 对象

Scala 中的 object(对象) 主要有以下几个作用:

因为 object 中的变量和方法都是静态的,所以可以用于存放工具类;

可以作为单例对象的容器;

可以作为类的伴生对象;

可以拓展类或特质;

可以拓展 Enumeration 来实现枚举。

3.1 工具类&单例&全局静态常量&拓展特质

这里我们创建一个对象 `Utils`,代码如下:

object Utils {/*
   *1. 相当于 Java 中的静态代码块 static,会在对象初始化时候被执行
   *   这种方式实现的单例模式是饿汉式单例,即无论你的单例对象是否被用到,
   *   都在一开始被初始化完成
   */val person = new Person// 2. 全局固定常量 等价于 Java 的 public static final val CONSTANT = "固定常量"// 3. 全局静态方法def low(word: String): String = {
    word.toLowerCase}
}

其中 Person 类代码如下:

class Person() {
  println("Person 默认构造器被调用")
}

新建测试类:

// 1.ScalaApp 对象扩展自 trait App
object ScalaApp extends App {// 2.验证单例
  println(Utils.person == Utils.person)// 3.获取全局常量
  println(Utils.CONSTANT)// 4.调用工具类
  println(Utils.low("ABCDEFG"))}// 输出如下:
Person 默认构造器被调用
true
固定常量
abcdefg

3.2 伴生对象

在 Java 中,你通常会用到既有实例方法又有静态方法的类,在 Scala 中,可以通过类和与类同名的伴生对象来实现。类和伴生对象必须存在与同一个文件中。

class Person() {private val name = "HEIBAIYING"def getName: String = {// 调用伴生对象的方法和属性
    Person.toLow(Person.PREFIX + name)}
}// 伴生对象
object Person {val PREFIX = "prefix-"def toLow(word: String): String = {
    word.toLowerCase}def main(args: Array[String]): Unit = {val person = new Person// 输出 prefix-heibaiying  
    println(person.getName)}
}

3.3 实现枚举类

Scala 中没有直接提供枚举类,需要通过扩展 `Enumeration`,并调用其中的 Value 方法对所有枚举值进行初始化来实现。

object Color extends Enumeration {// 1.类型别名,建议声明,在 import 时有用type Color = Value// 2.调用 Value 方法val GREEN = Value// 3.只传入 idval RED = Value(3)// 4.只传入值val BULE = Value("blue")// 5.传入 id 和值val YELLOW = Value(5, "yellow")// 6. 不传入 id 时,id 为上一个声明变量的 id+1,值默认和变量名相同val PINK = Value}

使用枚举类:

// 1.使用类型别名导入枚举类
import com.heibaiying.Color.Colorobject ScalaApp extends App {// 2.使用枚举类型,这种情况下需要导入枚举类def printColor(color: Color): Unit = {
    println(color.toString)}// 3.判断传入值和枚举值是否相等
  println(Color.YELLOW.toString == "yellow")// 4.遍历枚举类和值for (<- Color.values) println(c.id + ":" + c.toString)
}//输出
true
0:GREEN
3:RED
4:blue
5:yellow
6:PINK

相关文章:

Scala的类和对象

1. 初识类和对象 Scala 的类与 Java 的类具有非常多的相似性&#xff0c;示例如下&#xff1a; // 1. 在 scala 中&#xff0c;类不需要用 public 声明,所有的类都具有公共的可见性 class Person {// 2. 声明私有变量,用 var 修饰的变量默认拥有 getter/setter 属性private var…...

SQL中 <>(不等于)运算符只会匹配那些具有非空值的记录

0. 场景 idflag112null3 一张表的有有个varchar类型的flag字段,字段值有 null值/空值和 1。 flag为 1即表示逻辑删除,我想找出flag字段非 1 的所有情况: 一开始sql写法: select * from table where flag<>‘1’理想情况,结果集应该有2条记录(id为 2 和 3 的记录)实际情…...

冒泡排序(Java)

基本思想 比较前后相邻的二个数据&#xff0c;如果前面数据大于后面的数据&#xff0c;就将这二个数据交换。这样对数组的第 0 个数据到 N-1 个数据进行一次遍历后&#xff0c;最大的一个数据就“沉”到数组第N-1 个位置。如此循环 (N-1)次&#xff0c;每次循环需要比较的个数…...

k8s集群调度

目录 1、理论&#xff1a; 1.1、 概述&#xff1a; 1.2、Pod 是 Kubernetes 的基础单元&#xff0c;Pod 启动典型创建过程如下&#xff1a; 工作机制 **** 1.3、调度过程 *** 1.4、Predicate 有一系列的常见的算法可以使用&#xff1a; ** 1.5、 优先级由一系列键…...

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

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

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

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

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

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

第02章-变量与运算符

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

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

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

go测试库之apitest

&#x1f4e2;专注于分享软件测试干货内容&#xff0c;欢迎点赞 &#x1f44d; 收藏 ⭐留言 &#x1f4dd; 如有错误敬请指正&#xff01;&#x1f4e2;交流讨论&#xff1a;欢迎加入我们一起学习&#xff01;&#x1f4e2;资源分享&#xff1a;耗时200小时精选的「软件测试」资…...

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

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

jvm实践

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

redis-plus-plus访问REDIS集群

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

python把Word题库转成Excle题库

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

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

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

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

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

leetcode做题笔记210. 课程表 II

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

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

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

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

目录 摘要&#xff1a; 引言 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的阅读没有完全结束&#xff0c;和很多朋友一样&#xff0c;同时也因为工作的需要&#xff0c;最近开始转向LLM方向。 国内在大模型方面生态做的最好的&#xff0c;目前还是百度的PaddlePaddle&#xff0c;所以自己也就先从PP开始看起了。 众所周知…...

【网络协议】聊聊HTTPS协议

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

2023.11.2事件纪念

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

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

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

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

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

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

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

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

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

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

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

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

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

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&…...