基础提高(一):final的使用

Java的关键字final通常是指被它修饰的数据是不能被改变的,不想改变可能出于两种理由:设计或效率。以下是本文主要大纲:

  • final数据的使用
  • final参数的使用
  • final方法的使用
  • final类不能被继承

一、final数据

1.1 final修饰变量

  • 修饰的变量必须是基本数据类型:告知编译器这一块数据是不变的,这样可以在执行计算时,减少一些运行时的负担。
  • 论什么情况,final修饰的变量必须初始化。

主要代码如下:

package com.game.lll;
import java.util.Random;


public class FinalT {  
    private static Random random = new Random(20);  
    private final int VALUE_A = 10;  
    private static final int VALUE_B = 20;  
    public static final int VALUE_C = random.nextInt(10);  
    public  final int VALUE_D = random.nextInt(10);  
    public static void main(String[] args) {  
        FinalTest test = new FinalTest();  
        //test.VALUE_A = 5;//Error:不可以这样做  
        //test.VALUE_B  =21;//Error:不可以这样做  
          
        FinalT finalT = new FinalT();  
        FinalT finalT1 = new FinalT();  
          
        System.out.println("VALUE_C:"+VALUE_C);  
        System.out.println("VALUE_C:"+finalT.VALUE_C);  
        System.out.println("VALUE_C:"+finalT1.VALUE_C);  
        System.out.println("---------");  
        System.out.println("VALUE_D:"+finalT.VALUE_D);  
        System.out.println("VALUE_D:"+finalT1.VALUE_D);  
    }  
      
}

运行结果:

+------------------第13、15行编译错误-------------------------------+
The final field FinalTest.a cannot be assigned
+--------------------static final ---------------------------------+
VALUE_C:3
VALUE_C:3
VALUE_C:3
+-----------------------final--------------------------------------+
VALUE_D:6
VALUE_D:1
+------------------------------------------------------------------+

以上结果得出一点我们不能因为某数据时final的就认为在编译时可以知道它的值。在运行时使用随机生成的VALUE_C和VALUE_D说明了这一点。示例代码展示了final数值定义为static和非static的区别。此区别只有当数值在运行时内被初始化时才会显现,这是因为编译器对编译时数值一视同仁(并且他们可能因为优化而消失)。使用static final 的数值是不可以通过一个新的对象而改变的。这是因为在装载时已经被初始化,而不是每次创建新对象时初始化。

1.2 final用于引用

当使用final用于对对象的引用而不是基本类型时,对于基本类型final使数值恒定不变,而对于对象引用,final使引用恒定不变。一旦引用被初始化指向一个对象,就无法再把它改为指向另一个对象。然而,对象自身的数据是可以修改的。

package com.game.lll;

public class FinalTest {  
    private final int a = 10;  
      
    private final Value v1 = new Value(10);  
      
    public static void main(String[] args) {  
        FinalTest test = new FinalTest();  
        //test.a = 5;//不可以这样做  
        test.v1.value++;  
        //test.v1 = new Value(12);//Error:不可以这样做  
        System.out.println("对象内的数据:"+test.v1.value);  
    }  
  
    class Value{  
        int value;   
        public Value(int value) {  
            this.value = value;  
        }  
    }  
}  

运行结果:

+------------------------------------------------------------------+
对象内的数据:11
+------------------------------------------------------------------+

1.3 final用于数组

final用于数组和引用时一样的,数组只不过是另一种引用,对于这个变量的引用是不能被重新赋值,但是对象本身是可以修改的。代码如下:

package com.game.lll;    

public class FinalTest {    
    private final Value v1 = new Value(10);    
  
    private final int[] values = { 1, 2, 3, 4, 5, 6 };    
  
    public static void main(String[] args) {    
        FinalTest test = new FinalTest();    
        //test.a = 5;//不可以这样做    
        test.v1.value++;    
        //test.v1 = new Value(12);//Error:不可以这样做  
        test.values[0] = 100;//对象本身是可以修改的    
        for(int i = 0; i < test.values.length;i++){    
            System.out.println(test.values[i]++);//对于这个变量的引用是不能被重新赋值    
        }    
    }    
      
    class Value{    
        int value;     
        public Value(int value) {    
            this.value = value;    
        }    
    }    
}    

运行结果:

+------------------------------------------------------------------+
100
2
3
4
5
6
+------------------------------------------------------------------+

1.4 空白final

Java允许生成“空白final”,空白final是指被声明为final但又给未定初值的域。代码如下:

package com.game.lll;  
  
public class BlankFinal {  
    private final int a;  
      
    public BlankFinal(int i)  
    {  
        this.a = i;  
    }  
      
      
    public static void main(String[] args) {  
        BlankFinal blankFinal = new BlankFinal(5);  
        System.out.println(blankFinal.a);  
    }  
}  

虽然未对a直接赋值,但是在构造函数中对a进行了初始化。所以无论什么情况,都要确保final在使用前被初始化。

二、final参数

2.1final参数用在普通类中

Java允许在函数参数列表中使用final。不过在使用final后你不能更改参数引用所指向的对象,但是可以修改对象里面的值。

代码如下:

package com.game.lll;  
  
public class FinalArguments {  
  
    private Person person = new Person(3);  
      
    private void run(final Person p)  
    {  
        //p = new Person();//Error 不要这样做  
          
        System.out.println("修改前a="+p.a);  
        p.a = 6;  
        System.out.println("修改后a="+p.a);  
    }  
  
    public static void main(String[] args) {  
        FinalArguments arguments = new FinalArguments();  
        arguments.run(arguments.person);  
    }  
  
     class Person{  
         int a;  
        public Person(int a)  
        {  
            this.a = a;  
        }  
    }  
}

运行结果:


+------------------------------------------------------------------+
修改前a=3
修改后a=6
+------------------------------------------------------------------+
 

2.2 final参数用在匿名内部类中

如果定义一个匿名内部类,并且希望它使用一个在其外部定义的对象,那么编译器会要求其参数引用是final的。

package com.game.lll;  
  
/** 
 * final在匿名内部类的使用 
 * @author liulongling 
 * 
 */  
public class FinalInClass{  
      
    public abstract class AbstractFinalClass<Result> {  
        protected abstract Result onExecute(int a);  
  
        public Result execute(int a)  
        {  
            return this.onExecute(a);  
        }  
    }  
      
    private Integer a(final int value)  
    {  
        return new AbstractFinalClass<Integer>() {  
            @Override  
            protected Integer onExecute(int a) {  
                a = value;  
                return a;  
            }  
        }.execute(3);  
    }  
      
    public static void main(String[] args) {  
        FinalInClass inClass = new FinalInClass();  
        System.out.println(inClass.a(2));  
    }  
}  

三、final方法

3.1 final和private关键字

使用final方法的原因是把方法锁定,以防任何继承类修改它的含义。同时如果父类中的这个方法使用了private那么在子类也是不能重写这个方法。代码示例如下:

package com.game.lll;  
  
public class FinalOverriding {  
    class WithFinals {  
        private final void a()  
        {  
            System.out.println("WithFinals a");  
        }  
          
        private void b()  
        {  
            System.out.println("WithFinals b");  
        }  
    }  
      
      
    class OverridingFinal extends WithFinals{  
        private final void a()  
        {  
            System.out.println("OverridingFinal a");  
        }  
          
        private void b()  
        {  
            System.out.println("OverridingFinal b");  
        }  
    }  
      
    class OverridingFinal2 extends OverridingFinal{  
        private final void a()  
        {  
            System.out.println("OverridingFinal2 a");  
        }  
          
        private void b()  
        {  
            System.out.println("OverridingFinal2 b");  
        }  
    }  
      
    private OverridingFinal2 of2 = new OverridingFinal2();  
      
    public static void main(String[] args) {  
        FinalOverriding finalOverriding = new FinalOverriding();  
        finalOverriding.of2.a();  
        finalOverriding.of2.b();  
          
        OverridingFinal of = finalOverriding.of2;  
        of.a();  
        of.b();  
          
        WithFinals wf = of;  
        wf.a();  
        wf.b();  
          
          
    }  
}  

运行结果:

+------------------------------------------------------------------+
OverridingFinal2 a
OverridingFinal2 b
OverridingFinal a
OverridingFinal b
WithFinals a
WithFinals b
+------------------------------------------------------------------+

从结果上来看,子类OverridingFinal和OverridingFinal2都没有重写父类方法。在Java中类中private方法只能在供内部访问,所以子类是无法覆盖它。对private方法添加final和不加结果都是一样并没有任何意义。。下面对它做一些修改

第一步:将WithFinals的方法a()改为public

public final void a()  
{  
    System.out.println("WithFinals a");  
}  

第二步:将OverridingFinal的方法a()也改为public

public final void a()  
{  
   System.out.println("OverridingFinal a");  
}  

结果编译失败:

+------------------------------------------------------------------+
Cannot override the final method from FinalOverriding.WithFinals
+------------------------------------------------------------------+

将private修改为public后,结果证明使用final修饰的方法是不能被重写的。如果我们把OverridingFinal和OverridingFinal2的b()方法改为public呢?那么OverridingFinal2方法b将会覆盖OverridingFinal的方法b。

运行结果如下:

+------------------------------------------------------------------+
OverridingFinal2 a
OverridingFinal2 b
OverridingFinal a
OverridingFinal2 b
WithFinals a
WithFinals b
+------------------------------------------------------------------+

四、final类

当你将某个类定义为final class时,那么子类就无法继承该类。

本章参考文章和书籍
<<Thinking in Java>>

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,612评论 18 399
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,647评论 18 139
  • 一:java概述:1,JDK:Java Development Kit,java的开发和运行环境,java的开发工...
    ZaneInTheSun阅读 2,645评论 0 11
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,230评论 11 349
  • 【十四】 越怕发生的事情越容易发生,好像人总有一种很灵验的预感。 去往乾清宫的路上引路的小太监什么也不说,只是说皇...
    CUONA的中贞阅读 307评论 7 4