Java学习笔记18---final关键字修饰变量、方法及类

时间:2021-05-30 21:34:55

英语里final这个单词大家都知道是“最终的”意思,其实还有一个意思是“不可更改的”。在Java里,final关键字作“不可更改的”来解释更合适,即由final修饰的东西是“不可更改的”。final可以修饰变量、成员方法和类,还可以修饰方法的参数,但参数归根结底还是变量。下面是详细解释。

作者: 蝉蝉

请尊重作者劳动成果,转载请在标题注明“转载”字样,并标明原文链接:

http://www.cnblogs.com/chanchan/p/7936388.html

1.final可以修饰成员变量、局部变量

(1).final修饰的变量的值不能更改,即不能再次赋值,即使赋的新值与旧值一样也不可以。

如:

final int i = 1; //定义int型变量i,并赋初值为1

//i = 2; //wrong

//i = 1; //wrong

注释掉的这两行都不可以,都会出现“The final local variable i can not be assigned.It must be blank and not using a compound assignment.”的错误。

(2).final修饰的成员变量和局部变量需要在使用前赋值。

1).对成员变量而言,可以在定义时直接赋初值;

2).也可以先定义不赋初值,再在构造方法中对其赋值。

就是说,对象创建好后,该对象的final成员变量要保证是赋了初值的,早点(定义时赋值)、晚点(在构造方法里赋值)都没关系。

注:第二种情况,如果有多个构造方法就比较麻烦了,每个构造方法都要有赋值语句,否则会出现“The blank final field height may not have been initialized.”的错误。

      3).如果静态成员变量被修饰为final,那么该变量可看成是全局变量,即在类的整个加载期间,其值都不可变。

如:

static final String citizenship = "Chinese";

citizenship既是static又是final的,static说明它是属于整个类的,类加载时就分配内存了(参见笔记9),final说明其值不可变,即,位置固定内容也固定。

(3).把成员变量和局部变量从变量的类型上来分,

1).当变量为基本数据类型时,变量的值不可更改,如上面的变量i;

2).当变量为引用类型时,变量本身的值不可更改,即该变量不能指向其他的对象或数组;

该变量指向的对象或数组本身的内容是可以改变的

如:

final Person per = new Person(); //定义了一个Person类的对象引用per,并指向了new的对象

//per = new Person(); //重新创建一个Person类对象,并让per指向它,会出现与上面一样的错误,即final修饰的引用类型变量不能重新赋值

per.name = "me"; //per指向的对象本身的内容可以更改

为方便理解起见,请参考下面的内存图:

Java学习笔记18---final关键字修饰变量、方法及类

2.final可以修饰成员方法

(1).final修饰的成员方法不能被子类重写,即,当父类的方法为final时,子类不能与父类有方法名、参数类型、参数个数及参数顺序都一样的方法;父类方法为private时除外,详见下面的(3);但子类可以调用父类的final方法。

见下面的代码:

Person类的方法:
final void finalMethod() {
int i = 2;
System.out.println("finalMethod: i = " + i);
} 在TestMain中由Student类的对象引用调用:
public class TestMain {
public static void main(String[] args) {
Student stu = new Student(); stu.finalMethod();
}

输出结果为:finalMethod: i = 2

(2).访问权限为private的方法默认为final的,但子类不可以调用private的方法,关于访问权限修饰符的问题,详见笔记10

(3).当父类某方法为private final,子类的成员方法与父类的该成员方法重名但仅由final修饰时,这时算不算重写呢?

Person类的方法:
private final void priFinalMethod() {
System.out.println("Person:priFinalMethod");
} Student类的方法:
final void priFinalMethod() {
System.out.println("Student:priFinalMethod");
} TestMain类:
package human; public class TestMain {
public static void main(String[] args) {
Person per = new Person();
Student stu = new Student();
Person per1 = stu; // per.priFinalMethod();
stu.priFinalMethod();
// per1.priFinalMethod();
} Person类的main方法:
public static void main(String[] args) {
Person per = new Person();
Student stu = new Student();
Person per1 = stu; per.priFinalMethod();
stu.priFinalMethod();
per1.priFinalMethod();
}

TestMain类的输出是:

Student:priFinalMethod

    1).其中,TestMain类中注释掉的两行都提示该方法不可见;

    2).对于per来说,private的方法仅本类可见,在TestMain类中是不可见的,所以per是不能调用priFinalMethod方法的;

    3).对于per1来说,per1是指向Student对象的引用,per1只能调用Student中重写过的方法及Person类中的方法,由于这里仍然提示该方法不可见,结合2)可知,priFinalMethod方法是没被子类重写的,否则就可以调用了;

Person类的输出是:

Person:priFinalMethod
Student:priFinalMethod
Person:priFinalMethod

前两行好理解,最后一行,per1调用的是Person类中的priFinalMethod,进一步说明该方法未被子类重写;

否则,会优先调用子类的priFinalMethod方法的。

3.final可以修饰成员方法的参数

由final修饰的成员方法的参数也是不能更改的,其实参数就是变量,具体参见1即可。

这里还涉及到形参与实参的概念,具体大家自己了解吧。

4.final可以修饰类

由final修饰的类不能被子类继承,其成员方法也默认为final的,但成员变量是可以改变的,见下面代码:

package human;

public final class FinalClass {

    int i = 1;

    void test() {
System.out.println("FinalClass:test");
} public static void main( String[] args ) {
FinalClass ficl = new FinalClass(); System.out.println("ficl.i = " + ficl.i);
ficl.i = 2;
System.out.println("ficl.i = " + ficl.i);
}
}

输出结果为:

ficl.i = 1
ficl.i = 2

可见可以修改i的值。

附测试源码:

Person类:

package human;

public class Person {
//class Person{
String name;
int age;
String gender; //笔记18:final修饰成员变量,及成员变量为类引用时的情况
final int height = 160;
// final int height;
final EduBackground edu = new EduBackground(); public Person() {
//  final height = 160;
} //笔记18:final修饰局部变量、修饰成员方法、修饰方法的参数
//修饰局部变量时,局部变量的值不能改变
void finalLocal() {
// final int i = 1;
final int i; // i = 3;
final EduBackground edu = new EduBackground();
// edu = new EduBackground();
i = 1;
System.out.println("finalLocal: i = " + i);
} //修饰方法的参数时,参数不能被修改
void finalArgs(final int i) {
// i = 3;
System.out.println("finalArgs: i = " + i);
}
void finalArgs(final EduBackground edu) {
// edu = new EduBackground();
System.out.println("finalArgs: edu");
} //修饰成员方法时,成员方法不能被子类重写
final void finalMethod() {
int i = 2;
System.out.println("finalMethod: i = " + i);
}
private final void priFinalMethod() {
System.out.println("Person:priFinalMethod");
} public static void main(String[] args) {
Person per = new Person();
Student stu = new Student();
Person per1 = stu; per.priFinalMethod();
stu.priFinalMethod();
per1.priFinalMethod();
}
}

Student类:

package human;

public class Student extends Person {
String stuNumber;
int score; public Student() { } //笔记18:子类不能重写父类被final修饰的方法
// final void finalMethod() {
// int i = 2;
// System.out.println("finalMethod: i = " + i);
// }
final void priFinalMethod() {
System.out.println("Student:priFinalMethod");
} }
EduBackground类:
package human;

//public class EduBackground extends FinalClass {
public class EduBackground { String primarySchool;
String secondarySchool;
String juniorHSchool;
String seniorHSchool;
String university; public EduBackground() { }
}