Java学习笔记(29)--final关键字详解

来源:互联网 发布:企业数据接口 编辑:程序博客网 时间:2024/06/17 08:43

在Java中,final关键字可以用来修饰类、方法和变量(包括成员变量和局部变量)。下面就从这三个方面来了解一下final关键字的基本用法。

1.修饰类

当用final修饰一个类时,表明这个类不能被继承。也就是说,如果一个类你永远不会让他被继承,就可以用final进行修饰。final类中的成员变量可以根据需要设为final,但是要注意final类中的所有成员方法都会被隐式地指定为final方法。

示例代码:


注意:在使用final修饰类的时候,要注意谨慎选择,除非这个类真的在以后不会用来继承或者出于安全的考虑,尽量不要将类设计为final类。

2.修饰方法

该方法是最终方法,不能被重写

当一个类被继承,那么所有的非私有函数都将被继承,如果函数不想被子类继承并重写可以将该函数final修饰

当一个类中的函数都被修饰为final时,可以将类定义为final的。

示例代码:

[java] view plain copy
 print?
  1. class Father2{  
  2.     final void eat(){  
  3.         System.out.println("eating....");  
  4.     }  
  5. }  
  6.   
  7. class Son2 extends Father2{  
  8.     //该方法是最终方法,不能被重写  
  9.     void eat(){  
  10.         System.out.println("eating....");  
  11.     }  
  12. }  
  13. class Demo12 {  
  14.   
  15.       
  16.     public static void main(String[] args) {  
  17.   
  18.         // Demo12.PI=300; 无法为最终变量 PI 指定值  
  19.         System.out.println(Demo12.PI);  
  20.         Son2 s=new Son2();  
  21.         s.eat();  
  22.   
  23.     }  
注意:类的private方法会隐式地被指定为final方法。
3.修饰变量

 对于一个final变量,如果是基本数据类型的变量,则其数值一旦在初始化之后便不能更改;如果是引用类型的变量,则在对其初始化之后便不能再让其指向另一个对象。

举个例子:


下面的ob没有报错,但是我把那行i=1去掉,就会报错~

在了解了final关键字的基本用法之后,现在我们来看一下final关键字容易混淆的地方。

1.类的final变量和普通变量有什么区别?

 当用final作用于类的成员变量时,成员变量(注意是类的成员变量,局部变量只需要保证在使用之前被初始化赋值即可)必须在定义时或者构造器中进行初始化赋值,而且final变量一旦被初始化赋值之后,就不能再被赋值了。

   那么final变量和普通变量到底有何区别呢?下面请看一个例子:

[java] view plain copy
 print?
  1. public class Test {  
  2.     public static void main(String[] args)  {  
  3.         String a = "hello2";   
  4.         final String b = "hello";  
  5.         String d = "hello";  
  6.         String c = b + 2;   
  7.         String e = d + 2;  
  8.         System.out.println((a == c));  
  9.         System.out.println((a == e));  
  10.     }  
  11. }  
运行效果如图:

为什么第一个比较结果为true,而第二个比较结果为fasle。这里面就是final变量和普通变量的区别了,当final变量是基本数据类型以及String类型时,如果在编译期间能知道它的确切值,则编译器会把它当做编译期常量使用。也就是说在用到该final变量的地方,相当于直接访问的这个常量,不需要在运行时确定。这种和C语言中的宏替换有点像。因此在上面的一段代码中,由于变量b被final修饰,因此会被当做编译器常量,所以在使用到b的地方会直接将变量b 替换为它的  值。而对于变量d的访问却需要在运行时通过链接来进行。想必其中的区别大家应该明白了,不过要注意,只有在编译期间能确切知道final变量值的情况下,编译器才会进行这样的优化,比如下面的这段代码就不会进行优化:

[java] view plain copy
 print?
  1. public class Test {  
  2.     public static void main(String[] args)  {  
  3.         String a = "hello2";   
  4.         final String b = getHello();  
  5.         String c = b + 2;   
  6.         System.out.println((a == c));  
  7.    
  8.     }  
  9.        
  10.     public static String getHello() {  
  11.         return "hello";  
  12.     }  
  13. }  
效果如图:


2.被final修饰的引用变量指向的对象内容可变吗?

在上面提到被final修饰的引用变量一旦初始化赋值之后就不能再指向其他的对象,那么该引用变量指向的对象的内容可变吗?看下面这个例子:

[java] view plain copy
 print?
  1. public class Test {  
  2.     public static void main(String[] args)  {  
  3.         final Bus bus = new Bus();  
  4.         System.out.println(++bus.i);  
  5.     }  
  6. }  
  7.    
  8. class Bus {  
  9.     public int i = 0;  
  10. }  
效果如图:


这说明引用变量被final修饰之后,虽然不能再指向其他对象,但是它指向的对象的内容是可变的。

3.final和static

很多时候会容易把static和final关键字混淆,static作用于成员变量用来表示只保存一份副本,而final的作用是用来保证变量不可变。看下面这个例子:

[java] view plain copy
 print?
  1. public class Test {  
  2.     public static void main(String[] args)  {  
  3.         MyClass myClass1 = new MyClass();  
  4.         MyClass myClass2 = new MyClass();  
  5.         System.out.println("final:i "+myClass1.i);  
  6.         System.out.println("static:j "+myClass1.j);  
  7.         System.out.println("final:i "+myClass2.i);  
  8.         System.out.println("static:j "+myClass2.j);  
  9.    
  10.     }  
  11. }  
  12.    
  13. class MyClass {  
  14.     public final double i = Math.random();  
  15.     public static double j = Math.random();  
  16. }  

运行效果如图:


每次打印的两个j值都是一样的,而i的值却是不同的。从这里就可以知道final和static变量的区别了。

好了,final关键字就说到这里了~

原创粉丝点击