final
final可以修饰:属性,方法,类,局部变量(方法中的变量)
- final修饰的属性的初始化可以在编译期,也可以在运行期,初始化后不能被改变。
- final修饰的属性跟具体对象有关,在运行期初始化的final属性,不同对象可以有不同的值。
- final修饰的属性表明是一个常数(创建后不能被修改)。
- final修饰的方法表示该方法在子类中不能被重写,final修饰的类表示该类不能被继承。
- final在多线程中存在的重排序问题
static
static可以修饰:属性,方法,代码段,内部类(静态内部类或嵌套内部类)
- static修饰的属性的初始化在编译期(类加载的时候),初始化后能改变。
- static修饰的属性所有对象都只有一个值。
- static修饰的属性强调它们只有一个。
- static修饰的属性、方法、代码段跟该类的具体对象无关,不创建对象也能调用static修饰的属性、方法等
- static和“this、super”势不两立,static跟具体对象无关,而this、super正好跟具体对象有关。
- static不可以修饰局部变量。
static final
- static修饰的属性强调它们只有一个,final修饰的属性表明是一个常数(创建后不能被修改)。static final修饰的属性表示一旦给值,就不可修改,并且可以通过类名访问。
- static final也可以修饰方法,表示该方法不能重写,可以在不new对象的情况下调用。
从字节码的角度观察
public class Test {
public String a="1";
}
该文件编译后的class中的常量池如下
#1 = Methodref #5.#17 // java/lang/Object."<init>":()V
#2 = String #18 // 1
#3 = Fieldref #4.#19 // com/buckwallet/data/api/Test.a:Ljava/lang/String;
#4 = Class #20 // com/buckwallet/data/api/Test
#5 = Class #21 // java/lang/Object
#6 = Utf8 a
#7 = Utf8 Ljava/lang/String;
#8 = Utf8 <init>
#9 = Utf8 ()V
#10 = Utf8 Code
#11 = Utf8 LineNumberTable
#12 = Utf8 LocalVariableTable
#13 = Utf8 this
#14 = Utf8 Lcom/buckwallet/data/api/Test;
#15 = Utf8 SourceFile
#16 = Utf8 Test.java
#17 = NameAndType #8:#9 // "<init>":()V
#18 = Utf8 1
#19 = NameAndType #6:#7 // a:Ljava/lang/String;
#20 = Utf8 com/buckwallet/data/api/Test
#21 = Utf8 java/lang/Object
当添加final修饰时常量池多了
#8 = Utf8 ConstantValue
当添加Final static修饰时常量池少了
#19 = NameAndType #6:#7 // a:Ljava/lang/String;
说明fianl static修饰的字段编译后没有关系,可以通过下面代码验证
public class Test {
//输出结果
//1
public static void main(String[] args) {
System.out.println(A.a);
}
private static class A{
public final static String a="1";
static {
System.out.println("hello");
}
}
}
public class Test {
//输出结果
//hello
//1
public static void main(String[] args) {
System.out.println(A.a);
}
private static class A{
public static String a="1";
static {
System.out.println("hello");
}
}
}
当添加static修饰时常量池多了(类多初始方法)
#15 = Utf8 <clinit>
static {};
descriptor: ()V
flags: ACC_STATIC
Code:
stack=1, locals=0, args_size=0
0: ldc #2 // String 1
2: putstatic #3 // Field a:Ljava/lang/String;
5: return
LineNumberTable:
line 4: 0