Scala Class主构造器的参数有3种类型,val、var和普通类型(没有val、var标识),其中标识为val、var的参数的行为中《Programming in Scala》中说明的很清楚,普通类型则语焉不详。最近在写的一个类时碰到一个情况,类主构造器参数中有一个大对象,但在主构造器中会把这个对象的数据进行处理,转换成另外一种形式,而传进来的原始数据则不在需要,所以希望能被尽快释放掉,这样就担心Scala会不会像处理val、var那样在类里保持一个对该原始对象的引用,为了验证就把class文件用javap打开来看了一下。
假设有如下的一个类定义:
类有两个构造参数,nor在类的方法中会被用到,所以可以肯定会在类中定义一个field,而pa只在主构造器中用到,在类的其它方法中都没有用到,所以没有必要像nor一样定义成一个field的,那Scala编译器是不是这样处理的呢,我们用javap打开编译后的class文件看一下:
可以看到nor被转换成一个private final的变量nor,但并没有pa变量,而只有我们自己定义的*2之后的dpa变量及一个读取它的private int dpa()函数,可见,Scala已经做了优化,只在主构造器中使用,而在类定义中没有用到的构造参数会被优化掉。
假设有如下的一个类定义:
class Test(nor: Int, pa: Int) {
private val dpa = pa * 2
def normal = nor
def doublePa = dpa
}
类有两个构造参数,nor在类的方法中会被用到,所以可以肯定会在类中定义一个field,而pa只在主构造器中用到,在类的其它方法中都没有用到,所以没有必要像nor一样定义成一个field的,那Scala编译器是不是这样处理的呢,我们用javap打开编译后的class文件看一下:
public class Test implements scala.ScalaObject {
private final int nor;
private final int dpa
private int dpa();
public int normal();
public int doublePa();
public Test(int, int);
}
可以看到nor被转换成一个private final的变量nor,但并没有pa变量,而只有我们自己定义的*2之后的dpa变量及一个读取它的private int dpa()函数,可见,Scala已经做了优化,只在主构造器中使用,而在类定义中没有用到的构造参数会被优化掉。