final关键字的字面意思是最终的, 不可修改的. 这彷佛是一个看见名字就大概能知道怎么用的语法, 但你是否有深究过final在各个场景中的具体使用方法, 注意事项, 以及背后涉及的Java设计思想呢?html
// 代码示例
public static void main(String[] args) {
final Person p = new Person(20, "炭烧生蚝");
p.setAge(18); //能够修改p对象的数据
System.out.println(p.getAge()); //输出18
Person pp = new Person(30, "蚝生烧炭");
p = pp; //这行代码会报错, 不能经过编译, 由于p经final修饰永远指向上面定义的p对象, 不能指向pp对象.
}
复制代码
public static void main(String[] args) {
int n1 = 2019; //普通变量
final int n2 = 2019; //final修饰的变量
String s = "20190522";
String s1 = n1 + "0522"; //拼接字符串"20190512"
String s2 = n2 + "0522";
System.out.println(s == s1); //false
System.out.println(s == s2); //true
}
复制代码
首先要介绍一点: 整数-127-128是默认加载到常量池里的, 也就是说若是涉及到-127-128的整数操做, 默认在编译期就能肯定整数的值. 因此这里我故意选用数字2019(大于128), 避免数字默认就存在常量池中.java
总结: 这个例子想说明的是: 因为被final修饰的常量会在编译期进入常量池, 若是有涉及到该常量的操做, 颇有可能在编译期就已经完成.git
提示: 在JDK1.8之后, 经过内部类访问外部局部变量时, 无需显式把外部局部变量声明为final. 不是说不须要声明为final了, 而是这件事情在编译期间系统帮咱们作了. 可是咱们仍是有必要了解为何要用final修饰外部局部变量.编程
public class Outter {
public static void main(String[] args) {
final int a = 10;
new Thread(){
@Override
public void run() {
System.out.println(a);
}
}.start();
}
}
复制代码
javap -c .class文件的绝对路径
, 就能查看.class文件的反编译代码. 以上的Outter类通过编译产生两个.class文件, 分别是Outter.class和Outter$1.class
, 也就是说内部类会单独编译成一个.class文件. 下面给出Outter$1.class
的反编译代码.Compiled from "Outter.java"
final class forTest.Outter$1 extends java.lang.Thread {
forTest.Outter$1();
Code:
0: aload_0
1: invokespecial #1 // Method java/lang/Thread."<init>":()V
4: return
public void run();
Code:
0: getstatic #2 // Field java/lang/System.out:Ljava/io/PrintStream;
3: bipush 10
5: invokevirtual #3 // Method java/io/PrintStream.println:(I)V
8: return
}
复制代码
run()
方法反编译代码中的第3行:3: bipush 10
run()
方法执行过程当中是以压栈的形式存储到本地变量表中的, 也就是说在内部类打印变量a的值时, 这个变量a不是外部的局部变量a, 由于若是是外部局部变量的话, 应该会使用load
指令加载变量的值. 也就是说系统以拷贝的形式把外部局部变量a复制了一个副本到内部类中, 内部类有一个变量指向外部变量a所指向的值.
//原代码
public static void test(){
String s1 = "包夹方法a";
a();
String s2 = "包夹方法a";
}
public static final void a(){
System.out.println("我是方法a中的代码");
System.out.println("我是方法a中的代码");
}
//通过编译后
public static void test(){
String s1 = "包夹方法a";
System.out.println("我是方法a中的代码");
System.out.println("我是方法a中的代码");
String s2 = "包夹方法a";
}
复制代码