1. 简单类与无参方法

class Person {
  var age = 0 // 必须初始化字段
  def getAge() = age // 方法默认为公有的
}

备注

在Scala中,类并不声明为public. Scala源文件可以包含多个类,所有这些类都具有公有可见性.属性不声明默认为public.

使用类:

val p = new Person // 或者new Person()
p.age = 23
println(p.getAge()) // 23

调用无参方法时,可以写上圆括号,也可以不写:

p.getAge() // 23
p.getAge   // 23

2. 带getter和setter的属性

2.1 Java getter和setter

在Java类中,我们并不喜欢使用公有字段:

public class Person{
  public int age; // Java中不推荐使用这种方式
}

更倾向于使用getter和setter方法:

public class Person{
  private int age;
  public int getAge() {return age;}
  public void setAge(int age) {this.age = age;}
}

像这样的一对getter/setter通常被称为属性.我们会说Person类有一个age属性.

2.2 Scala getter和setter

在Scala中对每个字段都提供了getter和setter方法:

class Person{
  var age = 0
}

scala生成面向JVM的类,会生成一个私有的age字段以及相应的getter和setter方法.这两个方法都是公有的,因为我们没有将age声明为private.(对于私有字段而言,getter和setter方法也是私有的) 

在scala中getter和setter方法分别叫ageage_=.使用的时候如下:

val p = new Person
p.age = 21 // 调用p.age_=(21)
println(p.age) // 调用p.age()方法

备注

在scala中,getter和setter方法并非被命名为getXXX和setXXX,不过用意相同.

任何时候我们都可以自己重新定义getter和setter方法:

class Person {
  private var privateAge = 0
  def age = privateAge
  def age_= (newAge : Int): Unit = {
    if(newAge > 150){
      privateAge = 150
    }
    else if(newAge < 0){
      privateAge = 0
    }
  }
}

使用:

val p = new Person
p.age = -1;
println(p.age) // 0
p.age = 189
println(p.age) // 150

备注

Scala对每个字段生成getter和setter方法听上去有些恐怖,不过你可以控制这个过程:

  • 如果字段是私有的,则getter和setter方法也是私有的
  • 如果字段是val,则只有getter方法被生成
  • 如果你不需要任何的getter和setter方法,可以将字段声明为private[this]

2.3 Example

(1) 对于公有字段,getter和setter方法是公有的:

class Student {
  var age = 22
}

val stu = new Student
stu.age = 23
println(stu.age) // 23

(2) 对于私有字段,getter和setter方法是私有的:

class Student {
  private var age = 22
}

val stu = new Student
//stu.age = 23 // symbol age is inaccessible from this place
//println(stu.age) // symbol age is inaccessible from this place

(3) 如果字段是val,则只有getter方法被生成:

class Student {
  val age = 22
}

val stu = new Student
// stu.age = 23 // reassignment to val
println(stu.age) // 22

3. 只带getter的属性

如果只想需要一个只读的属性,有getter但没有setter,属性的值在对象构建完成之后就不再改变了,可以使用val字段:

class Student {
  val age = 22
}

Scala会生成一个私有的final字段和一个getter方法,但没有setter方法

4. 对象私有字段

在Scala中,方法可以访问该类的所有对象的私有字段:

class Counter {
  private var value = 0
  def increment(): Unit = {
    value += 1
  }
  // 对象可以访问另一个对象的私有字段
  def isLess (other : Counter) = value < other.value
}

之所以访问other.value是合法的,是因为other也是Counter对象,这与Java的private权限不一样.

Scala允许我们定义更加严格的访问限制,通过private[this]这个修饰符来实现:

private[this] var value = 0

这样other.value是不被允许访问的,这样以来Counter类只能访问当前对象的value字段,而不能访问同样是Counter类型的其他对象的字段.

Scala允许你将访问权限赋予指定得类,private[类名]可以定义仅有指定类的方法可以访问给定的字段.这里的类名必须是当前定义的类,或者是包含该类的外部类.

备注

对于类私有的字段(private),Scala会生成私有的getter和setter方法,但是对于对象私有的字段,不会生成getter和setter方法.

5. Bean属性

Scala对于你定义的字段提供了getter和setter方法,但是并不是Java工具所期望的.JavaBeans规范把Java属性定义为一对getXXX/setXXX方法.很多Java工具都依赖这样的命令习惯.

Scala给我们提供了@BeanProperty注解,这样就会字段生成我们期望的getXXX和setXXX方法:

class Student {
  @BeanProperty
  var age = 22
}

val stu = new Student
stu.setAge(25)
println(stu.getAge()) // 25

总结

scala字段 生成的方法 何时使用
val/var name 公有的name name_=(仅限var) 实现一个可以被公开访问并且背后是以字段形式保存的属性
@BeanProperty val/var name 公有的name getName() name_=(仅限var) setName() (仅限var) 与JavaBeans互操作
private val/var name 私有的name name_=(仅限var) 用于将字段访问限制在本类的方法.尽量使用private,除非真的需要一个公有属性
private[this] val/var name 用于将字段访问限制在同一个对象上调用的方法.不经常用
private[类名] val/var name 依赖于具体实现 将访问权限赋予外部类.不经常使用

6. 辅助构造器

Scala可以有任意多的构造器,不过,Scala有一个构造器比其他所有构造器都重要,就是主构造器,除了主构造器之外,类还有任意多的辅助构造器.其同Java中的构造器十分相似,只有两处不同:

  • 辅助构造器的名称为this
  • 每一个辅助构造器都必须以一个先前已定义的其他辅助构造器或主构造器的调用开始
class Person {
  private var name = ""
  private var age = 0

  def this (name : String){
    this() // 调用主构造器
    this.name = name
  }

  def this (name : String, age : Int){
    this(name) // 调用前一个辅助构造器
    this.age = age
  }
}

可以使用如下三种方式构造对象:

val p1 = new Person // 调用主构造器
val p2 = new Person("Bob") // 调用第一个辅助构造器
val p3 = new Person("Bob", 25) // 调用第二个辅助构造器

7. 主构造器

在Scala中,每个类都有主构造器.主构造器并不以this方法定义,而是与类定义交织在一起.

(1) 主构造器的参数直接放在类名之后

class Person(val name:String) {
  private var age = 0
  def this (name : String, age : Int){
    this(name) // 调用主构造器
    this.age = age
  }
}

主构造器的参数被编译成字段,其值被初始化成构造时传入的参数.上述示例中name和age为Person类的字段.

(2) 主构造器会执行类定义中的所有语句

class Person(val name:String) {
  println("constructed a person ...")
  private var age = 0

  def this (name : String, age : Int){
    this(name) // 调用主构造器
    this.age = age
  }
}

println语句是主构造器的一部分.每当有对象被构造出来时.上述代码就会被执行

(3) 通常可以在主构造器中使用默认参数来避免使用过多的辅助构造器

class Person(val name:String = "", val age: Int = 0) {
}

备注

如果类名之后没有参数,则该类具备一个无参主构造器.

[Scala]Scala学习笔记四 类的更多相关文章

  1. Scala入门学习笔记四--List使用

    前言 本篇将介绍一个和Array很相似的集合List,更多内容请参考:Scala教程 本篇知识点概括 List的构造 List与Array的区别 List的常用方法 List伴生对象的方法 ::和:: ...

  2. 初探swift语言的学习笔记四(类对象,函数)

    作者:fengsh998 原文地址:http://blog.csdn.net/fengsh998/article/details/29606137 转载请注明出处 假设认为文章对你有所帮助,请通过留言 ...

  3. Scala入门学习笔记三--数组使用

    前言 本篇主要讲Scala的Array.BufferArray.List,更多教程请参考:Scala教程 本篇知识点概括 若长度固定则使用Array,若长度可能有 变化则使用ArrayBuffer 提 ...

  4. Typescript 学习笔记四:回忆ES5 中的类

    中文网:https://www.tslang.cn/ 官网:http://www.typescriptlang.org/ 目录: Typescript 学习笔记一:介绍.安装.编译 Typescrip ...

  5. C#可扩展编程之MEF学习笔记(四):见证奇迹的时刻

    前面三篇讲了MEF的基础和基本到导入导出方法,下面就是见证MEF真正魅力所在的时刻.如果没有看过前面的文章,请到我的博客首页查看. 前面我们都是在一个项目中写了一个类来测试的,但实际开发中,我们往往要 ...

  6. IOS学习笔记(四)之UITextField和UITextView控件学习

    IOS学习笔记(四)之UITextField和UITextView控件学习(博客地址:http://blog.csdn.net/developer_jiangqq) Author:hmjiangqq ...

  7. java之jvm学习笔记四(安全管理器)

    java之jvm学习笔记四(安全管理器) 前面已经简述了java的安全模型的两个组成部分(类装载器,class文件校验器),接下来学习的是java安全模型的另外一个重要组成部分安全管理器. 安全管理器 ...

  8. python3.4学习笔记(四) 3.x和2.x的区别,持续更新

    python3.4学习笔记(四) 3.x和2.x的区别 在2.x中:print html,3.x中必须改成:print(html) import urllib2ImportError: No modu ...

  9. Java学习笔记——File类之文件管理和读写操作、下载图片

    Java学习笔记——File类之文件管理和读写操作.下载图片 File类的总结: 1.文件和文件夹的创建 2.文件的读取 3.文件的写入 4.文件的复制(字符流.字节流.处理流) 5.以图片地址下载图 ...

随机推荐

  1. RocEDU.阅读.写作《苏菲的世界》书摘(二)

    "迷信",多么奇怪的一个名词.如果你信基督教或伊斯兰教,这就叫"信仰",但如果你相信占星术或十三号星期五不吉利,就是迷信.谁有权利说别人相信的东西就是" ...

  2. linux top 各个标识的含义 详解

        top之前一直都是一知半解,今天周末加班,我的工作已经完成,在等同事吃饭,就把这个写下来. 第一行: top - 20:42:47 up 57 days,  1:25,  4 users,  ...

  3. LeetCode——Edit Distance

    Question Given two words word1 and word2, find the minimum number of steps required to convert word1 ...

  4. 【网络结构】VGG-Net论文解析

    目录 0. 论文链接 1. 概述 2. 网络结构 2.1 卷积核 2.2 池化核 2.3 全连接层 3. 训练 4. 测试 5. 其他 6.参考链接 @ 0. 论文链接 论文链接 1. 概述   VG ...

  5. String被设计成不可变和不能被继承的原因

    String是所有语言中最常用的一个类.我们知道在Java中,String是不可变的.final的.Java在运行时也保存了一个字符串池(String pool),这使得String成为了一个特别的类 ...

  6. Numpy 数据分析基础

    numpy 库 ndarray : numpy 的关键 a = np.array([1,2,3]) # 轴 a.ndim # 数组长度 a.size # 数组的型 a.shape # 类型 a.dty ...

  7. NPM Scripts -- onchange parallelshell

    Watch for changes to the styles.scss file and automatically compile it to the css file. Run multiple ...

  8. 从源码角度分析 Kotlin by lazy 的实现

    by lazy 的作用 延迟属性(lazy properties) 是 Kotlin 标准库中的标准委托之一,可以通过 by lazy 来实现. 其中,lazy() 是一个函数,可以接受一个 Lamb ...

  9. 获得Python脚本所在目录

    如何获得Python脚本所在目录的位置   On this page... (hide) 1.  以前的方法 2.  正确的方法 3.  实例说明   (Edit) 1.  以前的方法 如果是要获得程 ...

  10. 9.深入理解AbstractQueuedSynchronizer(AQS)

    1. AQS简介 在上一篇文章中我们对lock和AbstractQueuedSynchronizer(AQS)有了初步的认识.在同步组件的实现中,AQS是核心部分,同步组件的实现者通过使用AQS提供的 ...