《疯狂Java:突破程序员基本功的16课》读书笔记-第二章 对象与内存控制
Java内存管理分为两个方面:内存分配和内存回收。这里的内存分配特指创建Java对象时JVM为该对象在堆内存中所分配的内存空间。内存回收指的是当该Java对象失去引用,变成垃圾时,JVM的垃圾回收机制自动清理该对象,并回收该对象所占用的内存。由于JVM内置了垃圾回收机制回收失去引用的Java对象所占用的内存,所以很多Java开发者认为Java不存在内存泄漏,资源泄漏的问题。实际上这是一种错觉,Java程序依然会有内存泄漏。
由于JVM的垃圾回收机制由一条后台线程完成,本身也是非常消耗性能的,因此如果肆无忌惮地创建对象,让系统分配内存,那这些分配的内存都将由垃圾回收机制进行回收。这样做有两个坏处:
(1) 不断分配内存使得系统中可用内存减少,从而降低程序运行性能;
(2) 大量已分配内存的回收使得垃圾回收的负担加重,降低程序的运行性能。
2.1实例变量和类变量
Java程序的变量大体可分为成员变量和局部变量。其中局部变量可分为如下3类:
(1) 形参:在方法签名中定义的局部变量,由方法调用者为其赋值,随方法的结束而消亡。
(2) 方法内的局部变量:在方法内定义的局部变量,必须在方法内对其进行显式初始化。这种类型的局部变量从初始化完成后开始生效,随方法的结束而消亡。
(3) 代码块内的局部变量:在代码块内定义的局部变量,必须在代码块内对其进行显式初始化。这种类型的局部变量从初始化完成后开始生效,随代码块的结束而消亡。
局部变量的作用时间很短暂,他们都被存储在方法的栈内存中。
类体内定义的变量被成为成员变量(Field)。如果定义该成员变量时没有使用static修饰,该成员变量又被称为非静态变量或实例变量;如果使用了static修饰,则该成员变量又被成为静态变量或类变量。
(对于static关键字而言,从词义上看来,它是“静态”的意思。但从Java程序的角度来看,static的作用就是将实例成员变成类成员,static只能修饰在类里定义的成员部分,包括成员变量,方法,内部类,初始化块,内部枚举类。如果没有使用static修饰这些类里的成员,这些成员属于该类的实例;如果使用了static修饰,这些成员就属于类本身。从这个意义上看,static只能修饰类里的成员,不能修饰外部类,不能修饰局部变量,局部内部类)。
表面上看,Java类里定义成员变量时没有先后顺序,但实际上Java要求定义成员变量时必须采用合法的前向引用。示例如下:
public class ErrorDef { //下面代码将提示:非法向前引用 int num1 = num2 + 2; int num2 = 20; } public class ErrorDef2 { //下面代码将提示:非法向前引用 static int num1 = num2 + 2; static int num2 = 20; }
但如果一个是实例变量,一个是类变量,则实例变量总是可以引用类变量,因为类变量的初始化时机总是处于实例变量的初始化时机之前。示例如下:
public class RightDef { //下面代码完全正确 int num1 = num2 + 2; static int num2 = 20; }
2.1.1 实例变量和类变量的属性
使用static修饰的成员变量是类变量,属于该类本身;没有使用static修饰的成员变量是实例变量,属于该类的实例。在同一个JVM内,每个类只对应一个Class对象,但每个类可以创建多个Java对象。
由于同一个JVM内每个类只对应一个Class对象,因此同一个JVM内的一个类的类变量只需一块内存空间;但对于实例变量而言,该类每创建一次实例,就需要为实例变量分配一块内存空间。也就是说,程序中有几个实例,实例变量就需要几块内存空间。
下面程序可以很好地表现出实例变量属于对象,而类变量属于类的特性
class Person { String name; int age; static int eyeNum; public void info() { System.out.println("我的名字是:" + name + ", 我的年龄是:" + age); } } public class FieldTest { public static void main(String[] args) { //类变量属于该类本身,只要该类初始化完成,程序即可使用类变量。 Person.eyeNum = 2; //通过Person类访问eyeNum类变量 System.out.println("Person的eyeNum属性:" + Person.eyeNum); //创建第一个Person对象 Person p = new Person(); p.name = "猪八戒"; p.age = 300; //通过p访问Person类的eyeNum类变量 System.out.println("通过p变量访问eyeNum类变量:" + p.eyeNum); p.info(); //创建第二个Person对象 Person p2 = new Person(); p2.name = "孙悟空"; p2.age = 500; p2.info(); //通过p2修改Person类的eyeNum类变量 p2.eyeNum = 3; //分别通过p、p2和Person访问Person类的eyeNum类变量 System.out.println("通过p变量访问eyeNum类变量:" + p.eyeNum); System.out.println("通过p2变量访问eyeNum类变量:" + p2.eyeNum); System.out.println("通过Person类访问eyeNum类变量:" + Person.eyeNum); } }
执行完这段程序,内存情况如图:
当Person类初始化完成后,eyeNum类变量也随之初始化完成。
2.1.2 实例变量的初始化时机
从程序运行的角度来看,每次创建Java对象都会为实例变量分配内存空间,并对实例变量执行初始化。
从语法角度来看,程序可以在3个地方对实例变量执行初始化:
(1) 定义实例变量时指定初始值;
(2) 非静态初始化块中对实例变量指定初始值;
(3) 构造器中对实例变量指定初始值。
其中第1,2种方式比第3种方式更早执行,但他们的执行顺序与排列顺序相同。
下面程序示范了实例变量的初始化时机
class Cat { //定义name、age两个实例变量 String name; int age; //使用构造器初始化name、age两个实例变量 public Cat(String name , int age) { System.out.println("执行构造器"); this.name = name; this.age = age; } { System.out.println("执行非静态初始化块"); weight = 2.0; } //定义时指定初始值 double weight = 2.3; public String toString() { return "Cat[name=" + name + ",age=" + age + ",weigth=" + weight + "]"; } } public class InitTest { public static void main(String[] args) { Cat cat = new Cat("kitty" , 2); System.out.println(cat); Cat c2 = new Cat("Jerfield" , 3); System.out.println(c2); } }
当程序执行Cat cat = new Cat("kitty" , 2);时,创建第一个Cat对象,程序就会先执行Cat类的非静态初始化块,再调用该Cat类的构造器来初始化该Cat实例,内存分配如图:
从上图可以看出,该Cat对象的weight实例变量的值为2.3,而不是初始化块中指定的。这是因为,初始化块中指定初始值,定义weight时指定初始值,都属于对该实例变量执行的初始化操作,它们的执行顺序与它们在源程序中的排列顺序相同。在本程序中,初始化块中对weight的赋值位于定义weight语句之前,因此程序将先执行初始化块中的操作,执行完成后weight的值为2.0;然后再执行定义weight时指定的初始值,完成后weight为2.3,从这里看,初始化块中对weight所指定的初始化值每次都将被2.3所覆盖
2.1.3 类变量的初始化时机
从程序运行的角度来看,每JVM对一个Java类只初始化一次,因此Java程序每运行一次,系统只为类变量分配一次内存空间,执行一次初始化。
从语法角度来看,程序可以在2个地方对类变量执行初始化:
(1) 定义类变量时指定初始值;
(2) 静态初始化块中对类变量指定初始值
这两种方式的执行顺序与它们在源程序中排列顺序相同。下列程序示范了类变量的初始化时机。
public class StaticInitTest { //定义count类变量,定义时指定初始值。 static int count = 2; //通过静态初始化块为name类变量指定初始值 static { System.out.println("StaticInitTest的静态初始化块"); name = "Java编程"; } //定义name类变量时指定初始值 static String name = "疯狂Java讲义"; public static void main(String[] args) { //访问该类的两个类变量 System.out.println("count类变量的值:" + StaticInitTest.count); System.out.println("name类变量的值:" + StaticInitTest.name); } }
2.2 父类构造器
当创建任何Java对象时,程序总会先依次调用每个父类非静态初始化块,父类构造器执行初始化,最后才调用本类的非静态初始化块,构造器执行初始化。
2.2.1 隐式调用和显式调用
当调用某个类的构造器来创建Java对象时,系统总会先调用父类的非静态初始化块进行初始化。这个调用是隐式执行的,而且父类的静态初始化块总是会被执行。接着调用父类一个或多个构造器执行初始化,这个调用既可以是通过super进行显式调用,也可以是隐式调用。
当所有父类的非静态初始化块,构造器依次调用完成后,系统调用本类的非静态初始化块,构造器执行初始化,最后返回本类实例。
如果有本图的继承关系:
程序会按如下步骤进行初始化:
(1) 执行Object类非静态初始化块(如果有的话)。
(2) 隐式或显式调用Object类的一个或多个构造器执行初始化。
(3) 执行Parent类非静态初始化块(如果有的话)。
(4) 隐式或显式调用Parent类的一个或多个构造器执行初始化。
(5) 执行Mid类非静态初始化块(如果有的话)。
(6) 隐式或显式调用Mid类的一个或多个构造器执行初始化。
(7) 执行Sub类非静态初始化块(如果有的话)。
(8) 隐式或显式调用Sub类的一个或多个构造器执行初始化。
只要在程序创建Java对象,系统总是先调用最顶层父类的初始化操作,包括初始化块和构造器,然后依次向下调用所有父类的初始化操作,最终执行本类的初始化操作返回本类的实例。至于调用父类的哪个构造器执行初始化,则分为如下几种情况:
(1) 子类构造器执行体的第一行代码使用super显式调用父类构造器,系统将根据super调用里传入的实参列表来确定调用父类的哪个构造器;
(2) 子类构造器执行的第一行代码使用this显式调用本类中重载的构造器,系统将根据this调用里传入的实参列表来确定本类的另一个构造器(执行本类中另一个构造器时即进入第一种情况);
(3) 子类构造器执行提中既没有super调用,也没有this调用,系统将会在执行子类构造器之前,隐式调用父类无参数的构造器。
(super调用用于显式调用父类的构造器,this调用用于显式调用本类中另一个重载的构造器。Super调用和this调用都只能在构造器中使用,而且super调用和this调用都必须作为构造器的第一行代码,因此构造器中的super调用和this调用最多只能使用其中之一,而且最多只能调用一次)。
2.2.2 访问子类对象的实例变量
子类的方法可以访问父类的实例变量,这是因为子类继承父类就会获得父类的成员变量和方法;但父类的方法不能访问子类的实例变量,因为父类根本无从知道它将被哪个子类继承,它的子类将会增加怎样的成员变量。
但是,在极端的情况下,可能出现父类访问子类变量的情况,例子如下:
class Base { //定义了一个名为i的实例变量 private int i = 2; public Base() { this.display(); } public void display() { System.out.println(i); } } //继承Base的Derived子类 class Derived extends Base { //定义了一个名为i的实例变量 private int i = 22; //构造器,将实例变量i初始化为222 public Derived() { i = 222; } public void display() { System.out.println(i); } } public class Test { public static void main(String[] args) { //创建Derived的构造器创建实例 new Derived(); } }
程序main里只有new Derived();。它会调用Derived的构造器,因为Derived继承了Dase,而且Derived构造器里没有显式使用super来调用父类的构造器,因此系统将会自动调用Base中无参数的构造器来初始化。
最后程序输出是0。不是22,也不是222,这是怎么回事呢?
为了解释这个程序,首先需要澄清一个概念:Java对象是由构造器创建的吗?很多书籍,资料中会说,是的。
但实际情况是:构造器只负责对Java对象实例变量执行初始化(也就是赋初始值),在执行构造器代码之前,该对象所占的内存已经被分配下来,这些内存里值都是默认是空值。
当程序执行new Derived();时,系统会先为Derived对象分配内存空间。此时系统内存需要为这个Derived对象分配两块内存,它们分别用于存放Derived对象的两个i实例变量,其中一个属于Base类定义的i,一个属于Derived类定义的i,此时这2个i的值都是0。
接下来程序在执行Derived的构造器之前,首先会执行Base类的构造器。表面上看,Base的构造器内只有一行代码this.display();,但由于Base类定义了i时指定了初始值是2,因此经过编译处理后,该构造器应该包含如下两行代码。
i = 2;
this.display();
现在的问题就是此处的this代表谁?
Java的定义是,当this在构造器中时,this代表正在初始化的Java对象。此时的情况是:从源代码来看,此时的this位于Base构造器内,但这些代码实际放在Derived构造器内执行,是Derived构造器隐式调用了Base构造器的代码。由此可见,此时的this应该是Derived对象,而不是Base对象。但是这个this虽然代表Derived对象,但它却位于Base构造器中,它的编译时类型是Base,而它实际引用了一个Derived对象。
为了验证这一点,为Derived类增加一个简单的sub()方法,然后改造Base构造器
public Base() { System.out.println(this.i); this.display(); System.out.println(this.getClass()); //因为this的编译类型是Base,所以依然不能调用sub()方法 this.sub(); }
上面程序调用this.getClass()来获取this代表对象的类,将看到输出Derived类,这表明此时this引用代表是Derived对象,但程序调用sub()时,则无法通过编译,这是因为this的编译时类型是Base的缘故。
当变量的编译时类型和运行时类型不同时,通过该变量访问它引用的对象的实例变量时,该实例变量的值由声明该变量的类型决定。但通过该变量调用它引用的对象的实例方法时,该方法行为将由它实际所引用的对象来决定。因此,当程序访问this.i时,将会访问Base类中定义的i实例变量,也就是输出2;但执行this.display()时,则实际表现出Derived对象的行为,也就是输出Derived对象的i实例变量,即0。
2.2.3 调用被子类重写的方法
在访问权限允许的情况下,子类可以调用父类方法,这是因为子类继承父类会获得父类定义的成员变量和方法;但父类不能调用子类的方法,因此父类根本无从知道它将被哪个子类继承,它的子类将会增加怎样的方法。
但有一种特殊情况,当子类方法重写了父类方法之后,父类表面上只有调用属于自己的,被子类重写的方法,但随着执行context的改变,将会变成父类实际调用子类的方法。
实例如下:
class Animal { //desc实例变量保存对象toString方法的返回值 private String desc; public Animal() { //调用getDesc()方法初始化desc实例变量 this.desc = getDesc(); } public String getDesc() { return "Animal"; } public String toString() { return desc; } } public class Wolf extends Animal { //定义name、weight两个实例变量 private String name; private double weight; public Wolf(String name , double weight) { //为name、weight两个实例变量赋值 this.name = name; this.weight = weight; } //重写父类的getDesc()方法 @Override public String getDesc() { return "Wolf[name=" + name + " , weight=" + weight + "]"; } public static void main(String[] args) { System.out.println(new Wolf("灰太郎" , 32.3)); } }
这段代码会输出Wolf[name=null , weight=0.0],而不是Wolf[name=灰太狼 , weight=32.3],那我们赋的值哪去了?
理解这个程序的关键在于this.desc = getDesc();。表面上此处是调用父类中定义的getDesc()方法,但实际运行过程中,此处会变成调用被子类重写的getDesc()方法。
程序在执行new Wolf("灰太郎" , 32.3) 之前,系统会隐式执行父类无参数的构造器,也就是先执行this.desc = getDesc();但不再调用父类中定义的getDesc()方法,而是调用子类的getDesc()方法。此时还没有为name,weight赋值,所以会输出Wolf[name=null , weight=0.0]
通过上面分析可以看到,该程序产生这种输出的原因在于this.desc = getDesc();处调用的getDesc()方法是被子类重写过的方法。这样使得对Wolf对象的实例变量赋值的语句this.name = name;this.weight = weight;在getDesc()方法之后被执行,因此getDesc()方法不能得到Wolf对象的name,weight实例变量的值。
为了避免这种不希望看到的结果,应该避免在Animal类的构造器中调用被子类重写过的方法,因此将Animal类改成如下形式即可:
class Animal2 { public String getDesc() { return "Animal"; } public String toString() { return getDesc(); } }
经过改写的Animal2类不再提供构造器(系统会为止提供一个无参数的构造器),程序改由toString()方法来调用被重写的getDesc()方法。这就保证了对Wolf对象的实例变量赋值的语句his.name = name;this.weight = weight;在getDesc()方法之前被执行,从而使得getDesc()方法得到Wolf对象的name,weight实例变量的值。
如果父类构造器调用了被子类重写的方法,且通过子类构造器来创建子类对象,调用(不管是显式还是隐式)了这个父类构造器,就会导致子类的重写方法在子类构造器的所有代码之前被执行,从而导致子类的重写方法访问不到子类的实际变量值的情形。
2.3 父子实例的内存控制
继承是面向对象的3大特征之一,也是Java语言的重要特征,而父,子继承关系则是Java编程中需要重点注意的地方。下面将继续深入分析父子实例的内存控制。
2.3.1 继承成员变量和继承方法的区别
很多java书籍,资料都会介绍,当子类继承父类时,子类会获得父类中定义的成员变量和方法。当访问权限允许的情况下,子类可以直接访问父类中定义的成员变量和方法。这种介绍其实稍嫌笼统,因为Java继承中对成员变量和方法的处理是不同的,示例如下:
class Base { int count = 2; public void display() { System.out.println(this.count); } } class Derived extends Base { int count = 20; @Override public void display() { System.out.println(this.count); } } public class FieldAndMethodTest { public static void main(String[] args) { // //声明、创建一个Base对象 Base b = new Base(); // //直接访问count实例变量和通过display访问count实例变量 System.out.println(b.count); b.display(); //声明、并创建一个Derived对象 Derived d = new Derived(); //直接访问count实例变量和通过display访问count实例变量 System.out.println(d.count); d.display(); // //声明一个Base变量,并将Derived对象赋给该变量 Base bd = new Derived(); //(3) // //直接访问count实例变量和通过display访问count实例变量 System.out.println(bd.count); bd.display(); // //让d2b变量指向原d变量所指向的Dervied对象 Base d2b = d; //(4) //访问d2b所指对象的count实例变量 System.out.println(d2b.count); } }
在(3)这块代码中,直接通过db访问count实例变量,输出的将是Base(声明时的类型)对象的count实例变量的值;如果通过db来调用display()方法,该方法将表现出Derived(运行时类型)对象的行为方式。
程序(4)代码直接将d变量赋值给d2b变量,只是d2b变量的类型是Base。这意味着d2b和d两个变量指向同一个Java对象,因此如果在程序中判断d2b==d,将返回true。但是访问d.count时输出20,访问d2b.count时却输出2。这一点看上去很诡异:两个指向同一个对象的变量,分别访问它们的实例变量时却输出不同的值。这表明在d2b,d变量所指向的Java对象中包含了两块内存,分别存放值为2的count实例变量和值为20的count实例变量。
但不管是d变量,还是bd变量,d2b变量,只要它们实际指向一个Dervied对象,不管声明它们时用什么类型,当通过这些变量调用方法时,方法的行为总是表现出它们实际类型的行为;但如果通过这些变量来访问它们所指对象的实例变量,这些实例变量的值总是表现出声明这些变量所用类型的行为。由此可见,Java继承在处理成员变量和方法时是有区别的。
如果在子类重写了父类方法,就意味着子类里定义了方法彻底覆盖了父类里的同名方法,系统将不可能把父类里的方法转移到子类中。对于实例变量则不存在这样的现象,即使子类中定义了与父类完全同名的实例变量,这个实例变量依然不可能覆盖父类中定义的实例变量。
因为继承成员变量和继承方法之间存在这样的差别,所以对于一个引用类型的变量而言,当通过该变量访问它所引用的对象的实例变量时,该实例变量的值取决于声明该变量时类型;当通过该变量来调用它所引用的对象的方法时,该方法行为取决于它所实际引用的对象的类型。
2.3.2 内存中子类实例
看下面一段程序:
class Base { int count = 2; } class Mid extends Base { int count = 20; } public class Sub extends Mid { int count = 200; public static void main(String[] args) { //创建一个Sub对象 Sub s = new Sub(); //将Sub对象向上转型后赋为Mid、Base类型的变量 Mid s2m = s; Base s2b = s; //分别通过3个变量来访问count实例变量 System.out.println(s.count); System.out.println(s2m.count); System.out.println(s2b.count); } }
打印的结果是:
200
20
2
s,s2m,s2b,这3个变量所引用Java对象拥有3个count实例变量,需要3块内存存储它们。如图:
从上图可以看出,Sub对象不仅存储了它自身的count实例变量,还存储从Mid,Base两个父类那里继承到的count实例变量。但这3个count实例变量在底层是有区别的,程序通过Base型变量来访问该对象的count实例变量时,输出2;通过Mid新的变量来访问该对象的count实例变量时,输出20;当直接在Sub类中访问实例变量i时,输出200。
为了在Sub类中访问Mid类定义的count实例变量,可以在count之前加入super关键字为限定。例:super.count;。这样就可以访问到父类中定义的count实例变量。
系统内存中并不存在Mid和Base两个对象,程序内存中只有一个Sub对象,只是这个Sub对象中不仅保存了在Sub类中定义的所有实例变量,还保存了它的所有父类所定义的全部实例变量。
那super关键字的作用到底是什么?看下个程序:
class Fruit { String color = "未确定颜色"; //定义一个方法,该方法返回调用该方法的实例 public Fruit getThis() { return this; } public void info() { System.out.println("Fruit方法"); } } public class Apple extends Fruit { //重写父类的方法 @Override public void info() { System.out.println("Apple方法"); } //通过super调用父类的Info()方法 public void AccessSuperInfo() { super.info(); } //尝试返回super关键字代表的内容 public Fruit getSuper() { return super.getThis(); } String color = "红色"; public static void main(String[] args) { //创建一个Apple对象 Apple a = new Apple(); //调用getSuper()方法获取Apple对象关联的super引用 Fruit f = a.getSuper(); //判断a和f的关系 System.out.println("a和f所引用的对象是否相同:" + (a == f)); System.out.println("访问a所引用对象的color实例变量:" + a.color); System.out.println("访问f所引用对象的color实例变量:" + f.color); //分别通过a、f两个变量来调用info方法 a.info(); f.info(); //调用AccessSuperInfo来调用父类的info()方法 a.AccessSuperInfo(); } }
输出结果为:
a和f所引用的对象是否相同:true
访问a所引用对象的color实例变量:红色
访问f所引用对象的color实例变量:未确定颜色
Apple方法
Apple方法
Fruit方法
Java程序允许某个方法通过return this;返回调用该方法的Java对象,但不允许直接return super;甚至不允许直接将super当成一个引用变量使用。
通过上面程序可以看出:super关键字本身并没有引用任何对象,它甚至不能被当成一个真正的引用变量使用。主要有如下两个原因:
(1) 子类方法不能直接使用return super;但使用return this;返回调用该方法的对象是允许的。
(2) 程序不允许直接把super当成变量使用,例如,试图判断super和a变量是否引用同一个Java对象—super==a;但这条语句将引起编译错误。
至此,对父,子对象在内存中存储有了准确的结论:当程序创建一个子类对象时,系统不仅会为该类中定义的实例变量分配内存,也会为其父类中定义的所有实例变量分配内存,即使子类定义了与父类中同名实例变量。
如果在子类里定义了与父类中已有变量同名的变量,那么子类中定义的变量会隐藏父类中定义的变量。之一不是完全覆盖,因此系统为创建子类对象时,依然会为父类中定义的,被隐藏的变量分配内存空间。
为了在子类方法中访问父类中定义的,被隐藏的实例变量,或者为了在子类方法中调用父类中定义的,被覆盖的方法,可以通过super作为限定来修饰这些实例变量和实例方法。
2.3.3 父,子类的类变量
理解了上面介绍的父,子实例在内存中分配之后,接下来的父,子类的类变量基本与此类似。不同的是,类变量属于类本身,而实例变量则属于Java对象;类变量在类初始化阶段完成初始化,而实例变量则在对象初始化阶段完成初始化。
由于类变量本质上属于类本身,因此通常不会涉及父,子实例变量那样复杂的情形,但由于Java允许通过对象来访问类变量,因此也可以使用super作为限定来访问父类中定义的类变量。下面程序示范了这种用法:
class StaticBase { //定义一个count类变量 static int count = 20; } public class StaticSub extends StaticBase { //子类再定义一个count类变量 static int count = 200; public void info() { System.out.println("访问本类的count类变量:" + count); System.out.println("访问父类的count类变量:" + StaticBase.count); System.out.println("访问父类的count类变量:" + super.count); } public static void main(String[] args) { StaticSub sb = new StaticSub(); sb.info(); } }
要访问父类中定义的count类变量,有2中方式
(1) 直接使用父类的类名作为主调来访问count类变量。
(2) 使用super作为限定来访问count类变量。
2.4 final修饰符
final修饰符是Java语言中比较简单的一个修饰符,定义如下:
(1) final可以修饰变量,被final修饰的变量被赋初始值之后,不能对它重新赋值。
(2) final可以修饰方法,被final修饰的方法不能被重写。
(3) final可以修饰类,被final修饰的类不能派生子类。
只掌握定义是不够的,下面将从几个方面来分析final修饰符功能。
2.4.1 final修饰的变量
被final修饰的实例变量必须显式指定初始值,而且只能在如下3个位置指定初始值。
(1) 定义final实例变量时指定初始值。
(2) 在非静态初始化块中为final实例变量指定初始值。
(3) 在构造器中为final实例变量指定初始值。
对于普通实例变量,Java程序可以对它执行默认的初始化,也就是将实例变量的值指定为默认的初始值0或null,但对于final实例变量,则必须由程序员显式指定初始值。
下面程序示范了在3个地方对final实例变量进行初始化。
public class FinalInstanceVaribaleTest { //定义final实例变量时赋初始值 final int var1 = "疯狂Java讲义".length(); final int var2; final int var3; //在初始化块中为var2赋初始值 { var2 = "轻量级Java EE企业应用实战".length(); } //在构造器中为var3赋初始值 public FinalInstanceVaribaleTest() { this.var3 = "疯狂XML讲义".length(); } public static void main(String[] args) { FinalInstanceVaribaleTest fiv = new FinalInstanceVaribaleTest(); System.out.println(fiv.var1); System.out.println(fiv.var2); System.out.println(fiv.var3); } }
上面程序用了3种给final变量初始化的方式,但是经过编译器的处理,这3种方式都会被抽取到构造器中赋初始值。
对于final类变量而言,同样必须显式指定初始值,而且final类变量只能在2个地方初始值:
(1) 定义final类变量时指定初始值;
(2) 在静态初始化块中为final类变量指定初始值。
下面程序示范了2个地方对final类变量进行初始化
public class FinalClassVaribaleTest { //定义final类变量时赋初始值 final static int var1 = "疯狂Java讲义".length(); final static int var2; //在静态初始化块中为var2赋初始值 static { var2 = "轻量级Java EE企业应用实战".length(); } public static void main(String[] args) { System.out.println(FinalClassVaribaleTest.var1); System.out.println(FinalClassVaribaleTest.var2); } }
上面程序用了2种方法初始化,但是经过编译器的处理,这2种方式都会被抽取到静态初始化块中赋初始值。
当使用final修饰类变量时,如果定义该final类变量时指定了初始值,而且该初始值可以在编译时就被确定下来,系统将不会在静态初始化块中对该类变量赋初始值,而将是在类定义中直接使用该初始值代替该final变量。
对于一个使用final修饰的变量而言,如果定义该final变量时就指定初始值,而且这个初始值可以在编译时就确定下来,那么这个final变量将不再是一个变量,系统会将其当成“宏变量”处理。也就是说,所有出现该变量的地方,系统将直接把它当成对应的值处理。
2.4.2 执行“宏替换”的变量
对一个final变量,不管它是类变量,实例变量,还是局部变量,只要定义该变量时使用了final修饰符修饰,并在定义该final类变量时指定了初始值,而且该初始值可以在编译时就被确定下来,那么这个final变量本质上已经不再是变量,而是相当于一个直接量。
Final修饰符的一个重要用途就是定义“宏变量”。当定义final变量时就为该变量指定了初始值,而且该初始值可以在编译时就确定下来,那这个final变量本质上就是一个“宏变量”,编译器会把程序中所有用到该变量的地方直接替换成该变量的值。
Java会缓存所有曾经用过的字符串直接量。例如执行String a = “java”;语句之后,系统的字符串池中就会缓存一个字符串”java”;如果程序再次执行String b = “java”;系统将会让b直接指向字符串池中的”java”字符串,因此a==b将会返回true。
为了加深对final修饰符的印象,看如下程序
public class StringJoinTest { public static void main(String[] args) { String s1 = "疯狂Java"; String s2 = "疯狂" + "Java"; System.out.println(s1 == s2); //定义2个字符串直接量 String str1 = "疯狂"; String str2 = "Java"; //将str1和str2进行连接运算 String s3 = str1 + str2; System.out.println(s1 == s3); } }
S1是一个普通的字符串直接赋值“疯狂Java”,s2的值是两个字符串直接进行连接运算,由于编译器可以在编译阶段就确定s2的值为“疯狂Java”,所以系统会让s2直接指向字符串池中缓存中的“疯狂Java”字符串,所以s1==s2输出true。
对于S3而言,它的值有str1和str2进行连接运算后得到。由于str1和str2只是两个普通变量,编译器不会执行“宏替换”,因此编译器无法在编译时确定s3的值,不会让s3指向字符串池中缓存中的“疯狂Java”。由此可见,s1==s3输出false。
为了让s1==s3输出true很简单,只要编译器可以对str1,str2两个变量执行“宏替换”。及在str1和str2之前加上fianl,这样编译器即可在编译阶段就确定s3的值,就会让s3指向字符串池中缓存中的“疯狂Java”。
对于实例变量而言,除了可以在定义该变量时赋初始值之外,还可以在非静态初始化块,构造器中对它赋初始值,而且在这3个地方指定初始值的效果基本一样。但对于fianl实例变量而言,只有在定义该变量时指定初始值才会有“宏变量”的效果,在非静态初始化块,构造器中为final实例变量指定初始值则不会有这种效果,示例如下:
public class FinalInitTest { //定义3个final实例变量 final String str1; final String str2; final String str3 = "Java"; //str1、str2分别放在非静态初始化块、构造器中初始化 { str1 = "Java"; } public FinalInitTest() { str2 = "Java"; } //判断str1、str2、str3是否执行"宏替换" public void display() { System.out.println(str1 + str1 == "JavaJava"); System.out.println(str2 + str2 == "JavaJava"); System.out.println(str3 + str3 == "JavaJava"); } public static void main(String[] args) { FinalInitTest fit = new FinalInitTest(); fit.display(); } }
上面程序str3是true,前两个都是false,就说明了这个问题。
2.4.3 final方法不能被重写
如果父类中某个方法使用了final修饰符进行修饰,那么这个方法将不可能被它的子类访问到,因此这个方法也不可能被它的子类重写。例子如下:
class Base { private final void info() { System.out.println("Base的info方法"); } } public class FinalMethodTest extends Base { //这个info方法并不是覆盖父类方法。 // @Override public void info() { System.out.println("FinalMethodTest的Info方法"); } }
子类FinalMethodTest的info方法只是一个普通方法,并不是重写父类的方法,如果打开@Override的话会提示错误。
2.4.4 内部类中的局部变量
如果程序需要在匿名内部类中使用局部变量,那么这个局部变量必须使用final修饰符修饰。示例如下:
import java.util.*; interface IntArrayProductor { //接口里定义的product方法用于封装“处理行为” int product(); } public class CommandTest { //定义一个方法,该生成指定长度的数组,但每个数组元素由于cmd负责产生 public int[] process(IntArrayProductor cmd , int length) { int[] result = new int[length]; for (int i = 0; i < length ; i++ ) { result[i] = cmd.product(); } return result; } public static void main(String[] args) { CommandTest ct = new CommandTest(); final int seed = 5; //生成数组,具体生成方式取决于IntArrayProductor接口的匿名实现类 int[] result = ct.process(new IntArrayProductor() { public int product() { return (int)Math.round(Math.random() * seed); } } , 6); System.out.println(Arrays.toString(result)); } }
不仅匿名内部类,即使是普通内部类,在任何内部类中访问的局部变量都应该使用final修饰。
那为什么Java要求内部类访问的局部变量必须使用final修饰呢?
Java要求所有被内部类访问的局部变量都使用final修饰也是有其原因的:对于普通局部变量而言,它的作用域就是停留在该方法内,当方法执行结束,该局部变量也随之消失;但内部类则可能产生隐式的“闭包”,闭包将使得局部变量脱离它所在的方法继续存在。
下面程序是局部变量脱离它所在方法继续存在的例子:
public class ClosureTest { public static void main(String[] args) { //定义一个局部变量 final String str = "Java"; //在内部类里访问局部变量str new Thread(new Runnable() { public void run() { for (int i = 0; i < 100 ; i++ ) { //此处将一直可以访问到str局部变量 System.out.println(str + " " + i); //暂停0.1秒 try { Thread.sleep(100); } catch (Exception ex) { ex.printStackTrace(); } } } }).start(); //执行到此处,main方法结束 } }
上面程序定义了一个局部变量str。正常情况下,当程序执行完main后,他的声明周期就结束了,局部变量str的作用域也会随之结束。但只要新线程里的run方法没有执行完,匿名内部类的实例的生命周期就没有结束,将一直可以访问str局部变量的值,这就是内部类会扩大局部变量作用域的实例。
由于内部类可能扩大局部变量的作用域,如果再加上这个被内部类访问的局部变量没有使用final修饰,也就是说该变量的值可以随意改变,那将引起极大的混乱,因此Java编译器要求所有被内部类访问的局部变量必须使用final修饰符修饰。