一、初识类和对象

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

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

二、类

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的方法名分别叫做ageage_=

  1. class Person {
  2. private val name = "heibaiying"
  3. private var age = 12
  4. private[this] var birthday = "2019-08-08"
  5. // birthday只能被内部方法所访问
  6. def getBirthday: String = birthday
  7. }
  8. object Person {
  9. def main(args: Array[String]): Unit = {
  10. val person = new Person
  11. person.age = 30
  12. println(person.name)
  13. println(person.age)
  14. println(person.getBirthday)
  15. }
  16. }

解释说明:

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

  1. class Person {
  2. var name = ""
  3. private var age = ""
  4. }

依次执行下面编译命令:

  1. > scalac Person.scala
  2. > javap -private Person

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

  1. Compiled from "Person.scala"
  2. public class Person {
  3. private java.lang.String name;
  4. private java.lang.String age;
  5. public java.lang.String name();
  6. public void name_$eq(java.lang.String);
  7. private java.lang.String age();
  8. private void age_$eq(java.lang.String);
  9. public Person();
  10. }

2.3 @BeanProperty

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

  1. class Person {
  2. @BeanProperty var name = ""
  3. }
  4. object Person {
  5. def main(args: Array[String]): Unit = {
  6. val person = new Person
  7. person.setName("heibaiying")
  8. println(person.getName)
  9. }
  10. }

2.4 主构造器

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

  • 主构造器传入的参数默认就是val类型的,即不可变,你没有办法在内部改变传参;
  • 写在主构造器中的代码块会在类初始化的时候被执行,功能类似于Java的静态代码块static{}
  1. class Person(val name: String, val age: Int) {
  2. println("功能类似于Java的静态代码块static{}")
  3. def getDetail: String = {
  4. //name="heibai" 无法通过编译
  5. name + ":" + age
  6. }
  7. }
  8. object Person {
  9. def main(args: Array[String]): Unit = {
  10. val person = new Person("heibaiying", 20)
  11. println(person.getDetail)
  12. }
  13. }
  14. 输出:
  15. 功能类似于Java的静态代码块static{}
  16. heibaiying:20

2.5 辅助构造器

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

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

2.6 方法传参不可变

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

  1. class Person() {
  2. def low(word: String): String = {
  3. word="word" // 编译无法通过
  4. word.toLowerCase
  5. }
  6. }

三、对象

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

  • 因为object中的变量和方法都是静态的,所以可以用于存放工具类;
  • 可以作为单例对象的容器;
  • 可以作为类的伴生对象;
  • 可以拓展类或特质;
  • 可以拓展Enumeration来实现枚举。

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

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

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

其中Person类代码如下:

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

新建测试类:

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

3.2 伴生对象

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

  1. class Person() {
  2. private val name = "HEIBAIYING"
  3. def getName: String = {
  4. // 调用伴生对象的方法和属性
  5. Person.toLow(Person.PREFIX + name)
  6. }
  7. }
  8. // 伴生对象
  9. object Person {
  10. val PREFIX = "prefix-"
  11. def toLow(word: String): String = {
  12. word.toLowerCase
  13. }
  14. def main(args: Array[String]): Unit = {
  15. val person = new Person
  16. // 输出 prefix-heibaiying
  17. println(person.getName)
  18. }
  19. }

3.3 实现枚举类

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

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

使用枚举类:

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

参考资料

  1. Martin Odersky . Scala编程(第3版)[M] . 电子工业出版社 . 2018-1-1
  2. 凯.S.霍斯特曼 . 快学Scala(第2版)[M] . 电子工业出版社 . 2017-7

更多大数据系列文章可以参见个人 GitHub 开源项目: 程序员大数据入门指南

Scala 学习之路(八)—— 类和对象的更多相关文章

  1. Scala学习之路 (六)Scala的类、对象、继承、特质

    一.类 1.类的定义 scala语言中没有static成员存在,但是scala允许以某种方式去使用static成员这个就是伴生机制,所谓伴生,就是在语言层面上,把static成员和非static成员用 ...

  2. iOS学习10之OC类和对象

    本次是OC的第一节课,主要是学习和理解类与对象 1.面向对象 1> OOP(Object Oriented Programming)面向对象编程. 面向对象以事物为中心,完成某件事情都需要哪些事 ...

  3. scala学习笔记(3):类

    1 类 (1) scala把主构造函数放到类的定义中,让定义字段及相应方法变得简单起来. class People(age: Int, name: String) scala会自动将这个类变成publ ...

  4. Scala 编程(二)类和对象

    类,字段和方法 类是对象的蓝图.一旦定义了类,就可以用关键字new从类的蓝图里创建对象,类的定义: class ChecksumAccumulator { // class definition go ...

  5. Scala学习文档-样本类与模式匹配(match,case,Option)

    样本类:添加了case的类便是样本类.这种修饰符可以让Scala编译器自动为这个类添加一些语法上的便捷设定. //样本类case class //层级包括一个抽象基类Expr和四个子类,每个代表一种表 ...

  6. python cookbook第三版学习笔记十:类和对象(一)

    类和对象: 我们经常会对打印一个对象来得到对象的某些信息. class pair:     def __init__(self,x,y):         self.x=x         self. ...

  7. [javase学习笔记]-6.2 类与对象的关系

    这一节我们来看一下类与对象之间的关系. 我们学习java语言,目的就是用java语言对现实生活中的事物进行描写叙述.那么我们如何来描写叙述呢.这就引出了类,我们在实际实现时,是通过类的形式来体现的. ...

  8. Scala学习之路 (三)Scala的基本使用

    一.Scala概述 scala是一门多范式编程语言,集成了面向对象编程和函数式编程等多种特性.scala运行在虚拟机上,并兼容现有的Java程序.Scala源代码被编译成java字节码,所以运行在JV ...

  9. Scala 基础(4)—— 类和对象

    1. 类.字段和方法 Scala 用 class 关键字定义类,一旦定义好一个类,就可以使用 new 关键字创建对象. Scala 使用 new 调用无参构造器时,可以不使用 (),推荐不使用括号: ...

随机推荐

  1. Objective-C block实现代码分析

    block内部结构 让我们写一个block void exampleBlock() { // NSConcreteStackBlock int a = 1; __block int b = 2; in ...

  2. Linux下如何生成core dump 文件(解决segment fault段错误的问题)

    Linux下的C程序常常会因为内存访问等原因造成segment fault(段错误),如果此时core dump 的功能是打开的,在运行我们的可执行程序时就会生成一个名为core的文件,然后我们就可以 ...

  3. 如何控制WAP网站上输入框的默认键盘类型

    百度上对这样的资料介绍很多,基本上都和这个页面是一个意思 http://www.w3school.com.cn/html5/att_input_type.asp : 语法 <input type ...

  4. Mybatis动态建表

    在网上查了很多,都说Mybatis不支持动态建表,心凉了一节.还好找到这么一篇,找到了希望:http://www.zzzyk.com/show/ec5660d9cf1071b3.htm 经过在mysq ...

  5. AdaBoost算法原理及OpenCV实例

    备注:OpenCV版本 2.4.10 在数据的挖掘和分析中,最基本和首要的任务是对数据进行分类,解决这个问题的常用方法是机器学习技术.通过使用已知实例集合中所有样本的属性值作为机器学习算法的训练集,导 ...

  6. C#: Get current keyboard layout\input language

    原文 https://yal.cc/csharp-get-current-keyboard-layout/ On some occasions, you may want to get a " ...

  7. WPF中Polyline拐角的bug

    原文:WPF中Polyline拐角的bug       Polyline绘制折线在小角度(比如几度)的时候会出现不连续的现象,形成拐角的两条线段中有一段会超出,角度越小越明显.       问题如下图 ...

  8. React实现checkbox group多组选项和标签组显示的联动

    实现功能:勾选checkbox项,确定后,已勾选的checkbox项以tag标签的形式展示,tag标签可快捷删除. 实现过程: 使用React. 使用Ant Design的Checkbox.Tag组件 ...

  9. C#中将字符串转换成Md5值的方法

    原文:C#中将字符串转换成Md5值的方法 版权声明:有问题可联系博主QQ:15577969,大家一起相互交流和学习. https://blog.csdn.net/qq15577969/article/ ...

  10. 所有语言的Awesome(2)

    Curated list of awesome lists https://awesomeweekly.co https://github.com/sindresorhus/awesome ✨ Pre ...