final用法

final的作用随着所修饰的类型而不同

       1final修饰类中的属性或者变量

              无论属性是基本类型还是引用类型,final所起的作用都是变量里面存放的“值”不能变。

              这个值,对于基本类型来说,变量里面放的就是实实在在的值,如1,“abc”等。

              而引用类型变量里面放的是个地址,所以用final修饰引用类型变量指的是它里面的地址不能变,并不是说这个地址所指向的对象或数组的内容不可以变,这个一定要注意。

              例如:类中有一个属性是final Person p=new Person(“name”) 那么你不能对p进行重新赋值,但是可以改变p里面属性的值,p.setName(‘newName’);

              p是引用变量  它里面存的一个地址,不能改变  但是我们可以改变这个地址指向的对象的属性值;

              final修饰属性,声明变量时可以不赋值,而且一旦赋值就不能被修改了。对final属性可以在三个地方赋值:声明时、初始化块中、构造方法中。总之一定要赋值。      

      2final修饰类中的方法

             作用:可以被继承,但继承后不能被重写。

      3final修饰类

             作用:类不可以被继承。

思考一个有趣的现象:

       byte b1=1;

       byte b2=3;

       byte b3=b1+b2;//当程序执行到这一行的时候会出错,因为b1b2可以自动转换成int类型的变量,运算时java虚拟机对它进行了转换,结果导致把一个int赋值给byte—–出错

       如果对b1 b2加上final就不会出错

       final byte b1=1;

       final byte b2=3;

       byte b3=b1+b2;//不会出错,相信你看了上面的解释就知道原因了。

 

以上内容是转载于http://blog.csdn.net/linchunhua/article/details/5305452

以下为自己见解

我们知道一个定义为final的对象引用只能指向唯一一个对象(因为地址不能更改)但是一个对象的本身的值是可以改变的;

比如:

[java]

  1. Random rand=new Random();  
  2.   
  3. public final int A_1=rand.nextInt(10);  

这里A_1每次运行的结果都不同的;

 

为了将一个常量真正做到不可更改 可以将常量声明为static final;

 

[java]

  1. public static final int A_2=rand.nextInt(10);  

这里结果每次运行都是相同的;

static final形式,它在内存中为A_2开辟了一个恒定不变的区域,当再次实例化对象的时候,任然指向A_2这块内存区域,所以A_2保持不变;A_2是在

装载的时候被初始化,而不是每次创建新的对象时都被初始化

final方法

final定义方法不能被覆盖

注意:一个父类的某个方法定义了private修饰符  ,子类将无法访问  自然就无法覆盖

[java]

  1. //final 类  
  2. public class finalParent {  
  3.     private final void doit(){  
  4.         System.out.println(“父类.doit()”);  
  5.     }  
  6.     final void doit2(){  
  7.         System.out.println(“父类.doit2()”);  
  8.     }  
  9.     public void doit3(){  
  10.         System.out.println(“父类.doit3()”);  
  11.     }  
  12. }  

[java]

  1. public class finalFindMethod {  
  2.     public static void main(String[] args){  
  3.         Sub s=new Sub();  
  4.         s.doit();//结果是子类.doit()   
  5.         finalParent p=s;//向上转型  
  6.         p.doit2();  
  7.         p.doit3();  
  8.     }  
  9. }  
  10. //在父类定义了doit(),子类也定义了doit()看似是子类覆盖了父类的方法,但是覆盖必须可以满足一个对象向上转型为他的基本类型  
  11. //并且调用相同方法的这一条件;父类中带了private是主类的私有方法,子类只是新定义了一个方法而已,二不是覆盖  
  12. //final的方法不能被子类继承  
[java]

  1. public class Sub extends finalParent {  
  2.     final void doit() {  
  3.         System.out.println(“子类.doit()”);  
  4.     }  
  5.   
  6.     // final void doit2(){ //这里显示不能重写父类为final的方法  
  7.     //  
  8.     // }  
  9.     public void doit3() {  
  10.         System.out.println(“子类.doit3()”);  
  11.     }  
  12. }  
结果:
子类.doit()
父类.doit2()
子类.doit3()

 

//在父类定义了doit(),子类也定义了doit()看似是子类覆盖了父类的方法,但是覆盖必须可以满足一个对象向上转型为他的基本类型
//并且调用相同方法的这一条件;父类中带了private是主类的私有方法,子类只是新定义了一个方法而已,二不是覆盖
//final的方法不能被子类继承

标签