在上一节创建了一个scala类,如果没有更多的方法,scala类的定义还可以更简单一些,看一下下面这个CreditCard类的定义:

class CreditCard(val number: Int, var creditLimit: Int)

是的,只用一行就完成了类的定义,连大括号都不需要。

因为scala也是运行在JVM上,可以考虑以java的方式来看看编译后的类文件。查看的方式还是比较灵活的,可以使用JD-GUI,也可以使用javap –private CreditCard命令,还有一个在线反编译的网站ShowMyCode。反编译后的Java代码:

public class CreditCard {

  public int number() {
return number;
} public int creditLimit() {
return creditLimit;
} public void creditLimit_$eq(int x$1) {
creditLimit = x$1;
} public CreditCard (int number, int creditLimit) {
this.number = number;
this.creditLimit = creditLimit;
super ();
} private final int number;
private int creditLimit;
}

好长的java代码。首先scala默认将CreditCard类转换为了public。因为在CreditCard.scala中将number声明为val,所以在反编译生成的java代码中,number被定义为final。此外在编译后的代码中还可以看到一个构造器以及读写成员变量的方法。可以看到成员变量的getter和setter与我们在java中习惯使用的命名方式有些不一致。此外由于number有final修饰符,因此就没有它的setter方法。如果scala中的成员变量的定义符号既不是var也不是val,那Scala就会为之创建一个private字段以及private getter和setter方法,也因此不能在类外部访问这个成员变量了。

放到类定义中的所有可执行语句或表达式都会被视为类的构造器的组成部分。下面的代码就是一个示例:

class Sample {
println("You are constructing an instance of Sample")
} new Sample

在这段代码中先定义了一个类Sample,随后又创建了一个Sample类的实例,执行看一下:

在创建实例的时候输出了类定义中的print语句,因为这段print语句是构造器的一部分。

除了在主构造函数中提供成员变量,我们还可以在类里面定义其他字段、方法、零个或多个副构造函数。在下面这个类中在类里面定义了一个成员变量position、一个副构造函数this()、并且重写了toString()方法。

class Person(val firstName: String, val lastName: String) {

  private var position: String = _

  println("Creating " + toString())

  def this(firstName: String, lastName: String, positionHeld: String) {
this(firstName, lastName)
position = positionHeld
} override def toString(): String = {
firstName + "" + lastName + " holds " + position + " position "
}
} val john = new Person("John", "Smith", "Analyst")
println(john)
val bill = new Person("Bill", "Walker")
println(bill)

执行代码的结果如下:

稍稍关注下副构造函数的实现:如果有主构造函数的话,那么副构造函数的第一行必须是主构造函数或者其它副构造函数的调用。这一点倒是和java继承父类时有点相似。

此外还值得注意的就是成员变量position的定义,把这一行单独拎出来看看吧:

private var position: String = _

首先比较有趣的是初始化赋值,赋值是一个“_”——下划线。在这里“_”代表相应类型的默认值。对于Int,它的值是0;对于Double,它的值是0.0;对于String,它的值就是null。通过使用“_”,可以很方便地为var成员变量设置初始默认值。不过不能为val成员使用“_”,因为val成员不允许修改,所以必须显式指定初始值

通过查看Person.scala的字节码反编译出来的java类,可以看到scalac编译器为成员变量position默认设置了getter和setter方法(虽然没有按照我们习惯的JavaBean的方式进行设置)。scala中定义的成员变量的可见性在反编译出来的Java代码中由getter和setter方法的访问权限来控制。

如果更喜欢传统的JavaBean式的注解,可以在成员变量定义时添加注解@BeanProperty:

@BeanProperty  var position: String = _

声明前记得导入注解。不过这也有一点限制:此时成员变量不可再声明为private。而且这样做只是会再额外生成两个JavaBean式的getter和setter,原来的getter和setter也会继续保留。这可以在编译Person类以后再用javap –private验证一下:

就是这样。

#######

scala学习手记12 - 字段、方法和构造函数的更多相关文章

  1. scala学习手记34 - trait方法的延迟绑定

    trait的方法的延迟绑定就是先混入的trait的方法会后调用.这一点从上一节的实例中也可以看出来. 下面再来看一个类似的例子: abstract class Writer { def write(m ...

  2. scala学习手记20 - 方法返回类型推断

    除了推演变量的类型,scala也会推演方法的返回类型.不过这里有一处需要注意:方法返回类型的推演依赖于方法的定义方式.如果用等号"="定义方法,scala就会推演方法返回类型:否则 ...

  3. scala学习手记16 – scala中的static

    前面两节学了scala的对象和伴生对象,这两个在使用的时候很有些java的静态成员的意思. scala中没有静态字段和静态方法.静态成员会破坏scala所支持的完整的面向对象模型.不过可以通过伴生对象 ...

  4. scala学习手记13 - 类继承

    在scala里,类继承有两点限制: 重写方法需要使用override关键字: 只有主构造函数才能往父类构造函数中传参数. 在java1.5中引入了override注解,但不强制使用.不过在scala中 ...

  5. scala学习手记2 - scala中的循环

    先来看一段Java中的循环: for (int i = 1; i < 4; i++) { System.out.print(i + ","); } 毫无疑问,scala可以让 ...

  6. scala学习手记38 - 方法命名约定和for表达式

    方法命名约定 之前在学习<运算符重载>一节时曾经说过一个方法命名约定:方法的第一个字符决定了方法的优先级.现在再说另一个命名约定:如果方法以冒号(:)结尾,则调用目标是运算符后面的实例. ...

  7. scala学习手记10 - 访问修饰符

    scala的访问修饰符有如下几个特性: 如果不指定访问修饰符,scala默认为public: 较之Java,scala对protected的定义更加严格: scala可以对可见性进行细粒度的控制. s ...

  8. scala学习手记28 - Execute Around模式

    我们访问资源需要关注对资源的锁定.对资源的申请和释放,还有考虑可能遇到的各种异常.这些事项本身与代码的逻辑操作无关,但我们不能遗漏.也就是说进入方法时获取资源,退出方法时释放资源.这种处理就进入了Ex ...

  9. scala学习手记23 - 函数值

    scala的一个最主要的特性就是支持函数编程.函数是函数编程中的一等公民:函数可以作为参数传递给其他函数,可以作为其他函数的返回值,甚至可以在其它函数中嵌套.这些高阶函数称为函数值. 举一个简单的例子 ...

随机推荐

  1. 只有ReflectionOnlyLoadFrom才可以拯救与GAC冲突的强命名程序集

    先说结论,如果有两个拥有相同程序集名称的强命名程序集,一个在GAC里,一个不在.怎样动态加载那个不在GAC里的程序集?答案就是只有Assembly.ReflectionOnlyLoadFrom才可以加 ...

  2. 使用synergyc共享键鼠

    通常情况下我们经常同时操作两台或者多台电脑, 这样就会存在多个键盘鼠标来回切换的问题. 那么我们主要的目标就是怎么在多个电脑上共享一套键盘鼠标,而且可以轻松的来回切换呢. 网上有很多的解决方案,这里我 ...

  3. IE11 Enterprise Mode Template missing from GPMC

    IE11 Enterprise Mode Template missing from GPMC     Reason:You have not copied the new IE11 Enterpri ...

  4. convention over configuration 约定优于配置 按约定编程 约定大于配置 PEP 20 -- The Zen of Python

    为什么说 Java 程序员必须掌握 Spring Boot ?_知识库_博客园 https://kb.cnblogs.com/page/606682/ 为什么说 Java 程序员必须掌握 Spring ...

  5. 我的Java开发学习之旅------>Java利用Comparator接口对多个排序条件进行处理

    一需求 二实现Comparator接口 三验证排序结果 验证第一条件首先按级别排序级别最高的排在前面 验证第二条如果级别相等那么按工资排序工资高的排在前面 验证第三条如果工资相当则按入职年数排序入职时 ...

  6. 静默安装oracle 11g及参数配置优化详解

    一.安装前准备工作1.修改主机名#vi /etc/hosts   //并添加内网IP地址对应的hostname,如下127.0.0.1           localhost::1           ...

  7. C#简单实现动态数据生成Word文档并保存

    今天正好有人问我,怎么生成一个报表式的Word文档. 就是文字的样式和位置相对固定不变,只是里面的内容从数据中读取. 我觉得类似这种的一般用第三方报表来做比较简便.但既然要求了Word,只好硬着头皮来 ...

  8. linux环境变量配置文件

    环境变量配置文件中主要是定义对系统操作环境生效的系统默认环境变量,如PATH等.当你登陆Linux系统启动一个bash shell时,默认情况下bash会几个文件中查找命令,bash检查的启动文件取决 ...

  9. 人性化的Form(django)

    django中的Form一般有两种功能: 输入html 验证用户输入 html: <!DOCTYPE html> <html lang="en"> < ...

  10. Oracle 在64位机器上使用plSQL连接Oracle的问题(SQL*Net not properly installed)

    问题: 在64位机器上了64位的oracle客户端. 然后装上PL/SQL Developer,但是连接oracle老报这个错: Initialization error      SQL*Net n ...