Java final、static 关键字

时间:2021-12-27 17:09:56

问:谈谈 Java 中 final、finally、finalize 的区别?

 

答:这道题其实没有任何意义,无非就是考察开发者有没有区分这几个关键字的含义,仅仅关联是因为长得像而已。

final 是一个修饰符,如果一个类被声明为 final 则其不能再派生出新的子类,所以一个类不能既被声明为 abstract 又被声明为 final 的;将变量或方法声明为 final 可以保证它们在使用中不被改变(对于对象变量来说其引用不可变,即不能再指向其他的对象,但是对象的值可变),被声明为 final 的变量必须在声明时给定初值,而在以后的引用中只能读取不可修改,被声明为 final 的方法也同样只能使用不能重载。使用 final 关键字如果编译器能够在编译阶段确定某变量的值则编译器就会把该变量当做编译期常量来使用,如果需要在运行时确定(譬如方法调用)则编译器就不会优化相关代码;将类、方法、变量声明为 final 能够提高性能,这样 JVM 就有机会进行估计并进行优化;接口中的变量都是 public static final 的。

finally 用来在异常处理时提供块来执行任何清除操作,如果抛出一个异常,则相匹配的 catch 子句就会执行,然后控制就会进入 finally 块。

finalize 是一个方法名,Java 允许使用 finalize() 方法在垃圾收集器将对象从内存中清除出去之前做必要的清理工作,这个方法是由垃圾收集器在确定这个对象没有被引用时对这个对象调用的,它是在 Object 类中定义的,因此所有的类都继承了它,子类覆盖 finalize() 方法以整理系统资源或者执行其他清理工作,finalize() 方法在垃圾收集器删除对象之前对这个对象调用的。

问:java 中 static、final、static final 的区别是什么?

 

答:final 可以修饰属性、方法、类、局部变量(方法中的变量),修饰属性的初始化可以在编译期,也可以在运行期,初始化后不能被改变;修饰的属性表明是一个常数;修饰方法表示方法不能在子类中被重写;修饰类表示类不能被继承。

static 可以修饰属性、方法、代码段、内部类(静态内部类或嵌套内部类),修饰属性的初始化在编译期(类加载的时候),初始化后可以被修改值;修饰的属性、方法、代码段跟该类的具体对象无关,不创建对象也能调用 static 修饰的属性、方法等;static 不可以修饰局部变量。

static final(或者 final static)是组合修饰,static 修饰的属性强调它们只有一个,final 修饰的属性表明是一个常数(创建后不能被修改),static final 修饰的属性表示一旦给值就不可修改并且可以通过类名访问,static final 也可以修饰方法,表示该方法不能重写,可以在不 new 对象的情况下调用。

问:下面程序的有问题吗,结果是什么?

class Test {

   public static String foo(){

       System.out.println("foo called.");

       return "return called.";

   }

}

public class Demo {

   public static void main(String[] args) {

       Test obj = null;

       System.out.println(obj.foo());

   }

}

答:没有问题,运行结果如下:

foo called.

return called.

因为 jvm 内存里有栈区、堆区,栈区主要用来存放基础类型数据和局部变量,堆区主要存放 new 出来的对象,在堆区又有一个叫做方法区的内存区域用来存放常量、static 变量和 static 方法、还有类的信息,static 的变量和方法不依赖对象,即使对象没有创建,在类加载的时候已经存在信息了(Test 在声明时就被加载了),jvm 识别出是 static 方法就直接调用了在方法区内存里的方法,没有报空指针异常。

问:下面程序的运行结果是什么?为什么

public class Test {

   public static void main(String[] args) {

       String a = "hello2";

       final String b = "hello";

       String c = b + 2;

       String d = "hello";

       String e = d + 2;

       System.out.println((a == c));

       System.out.println((a == e));

   }

}

答:运行结果如下:

true

false

因为当 final 变量是基本数据类型以及 String 类型时如果在编译期间能知道它的确切值则编译器会把它当做编译期常量使用,也就是说在用到该 final 变量的地方相当于直接访问了这个常量,不需要在运行时确定,所以上面代码中由于变量 b 被 final 修饰从而被当做编译器常量,故在使用到 b 的地方会直接将变量 b 替换为它的值,而对于变量 d 的访问却需要在运行时通过链接来进行。