java常量的final和static final

最近看JVM,对这个有点新的理解,记下来

static+final
静态常量,编译期常量,编译时就确定值。(Java代码执行顺序,先编译为class文件,在用虚拟机加载class文件执行)
放于方法区中的静态常量池。
在编译阶段存入调用类的常量池中
如果调用此常量的类不是定义常量的类,那么不会初始化定义常量的类,因为在编译阶段通过常量传播优化,已经将常量存到调用类的常量池中了
class ConstC{
    static{
        System.out.println("ConstC init!");
    }
    public ConstC(){
        System.out.println("ConstC ");
    }
    public static final String HELLO = "hello world!";
}
public class NotInit {
 
    public static void main(String[] args) {
        //经过编译优化,静态常量HELLO已经存到NotInit类自身常量池中,不会加载ConstC
        System.out.println(ConstC.HELLO);
    }
 
}
//输出:hello world!


final
常量,类加载时确定或者更靠后。
当用final作用于类的成员变量时,成员变量(注意是类的成员变量,局部变量只需要保证在使用之前被初始化赋值即可)必须在定义时或者构造器中进行初始化赋值
对于一个final变量,如果是基本数据类型的变量,则其数值一旦在初始化之后便不能更改;
如果是引用类型的变量,则在对其初始化之后便不能再让其指向另一个对象。但是它指向的对象的内容是可变的
 

1、如果常量引用的是一个固定值,那么加不加static的效果是一样的,除非声明为public的公共常量,需要加上static,可以直接调用。

2、如果常量引用的是一个函数,或者其他对象的赋值,那么就必须加上static了,要不,你就为莫名其妙的错误而头大了

final修饰的变量一旦初始化就不能被改变,但每次创建对象都会被初始化 
static final修饰的变量只能被初始化一次,而且一旦初始化就不能被改变

猜你喜欢

转载自blog.csdn.net/qq_29951485/article/details/88087951
今日推荐