亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java關(guān)鍵詞final示例解讀

 更新時(shí)間:2022年12月20日 10:53:32   作者:拿了桔子跑-范德依彪  
這篇文章主要介紹了Java關(guān)鍵詞final解讀,本文通過(guò)示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下

1 final基本用法

final:“這是無(wú)法改變的"
final可以修飾:變量、參數(shù)、方法、類(lèi)

1.1 final修飾變量

修飾變量(變量、局部變量),當(dāng)變量類(lèi)型為:

  • 基本類(lèi)型,一旦被賦值,該值不能被改變。
  • 引用類(lèi)型,一旦引用被初始化指向一個(gè)對(duì)象,就不能指向別的對(duì)象,但對(duì)象內(nèi)容可以被修改
  • 數(shù)據(jù)類(lèi)型:數(shù)組也是引用類(lèi)型

分析以下代碼:

import java.util.Random;
class Value {
    int i; // Package access
    public Value(int i) { this.i = i; }
}
public class FinalData {
    private static Random rand = new Random(47);
    private String id;
    public FinalData(String id) { this.id = id; }
    //編譯時(shí)常量
    private final int valueOne = 9;
    private static final int VALUE_TWO = 99;
    public static final int VALUE_THREE = 39;
    //非編譯時(shí)常量
    private final int i4 = rand.nextInt(20);
    static final int INT_5 = rand.nextInt(20);
    private Value v1 = new Value(11);
    private final Value v2 = new Value(22);
    private static final Value VAL_3 = new Value(33);
    // Arrays:
    private final int[] a = { 1, 2, 3, 4, 5, 6 };
    public String toString() {
        return id + ": " + "i4 = " + i4 + ", INT_5 = " + INT_5;
    }

    public static void main(String[] args) {
        FinalData fd1 = new FinalData("fd1");
        //! fd1.valueOne++; // Error: can't change value
        fd1.v2.i++; // OK:引用指向的對(duì)象內(nèi)容可變
        fd1.v1 = new Value(9); // OK :非final,引用可變
        for(int i = 0; i < fd1.a.length; i++)
            fd1.a[i]++; // Object isn't constant!
        //! fd1.v2 = new Value(0); // Error: final引用不可變
        //! fd1.VAL_3 = new Value(1); //Error: final引用不可變
        //! fd1.a = new int[3];
        System.out.println(fd1);
        System.out.println("Creating new FinalData");
        FinalData fd2 = new FinalData("fd2");
        System.out.println(fd1);
        System.out.println(fd2);
    }
}
/* 運(yùn)行結(jié)果:
fd1: i4 = 15, INT_5 = 18
Creating new FinalData
fd1: i4 = 15, INT_5 = 18
fd2: i4 = 13, INT_5 = 18
*///:~

說(shuō)明:

  • valuOne和VALUE_TWO:都是編譯期常量,無(wú)重大區(qū)別。
  • VAL_THREE:典型的對(duì)常量定義的方式:定義為public,則可以被用于包之外;定義為static,則強(qiáng)調(diào)只有一份;定義為final,則說(shuō)明它是一個(gè)常量。注意這種類(lèi)型常量的命名方式(大寫(xiě)和下劃線)
  • i4和INT_ 5:final變量不代表編譯時(shí)可知它的值,可以在運(yùn)行時(shí)初始化值。例如在運(yùn)行時(shí)使用隨機(jī)生成的數(shù)值來(lái)初始化
  • v1、v2、VAL_3 說(shuō)明final引用的特征
  • 特別注意:INT_5:不可以通過(guò)創(chuàng)建第二個(gè)FinalData對(duì)象而加以改變。因?yàn)樗?strong>static的,在裝載時(shí)已被初始化,而不是每次創(chuàng)建新對(duì)象時(shí)都初始化。

1.2 final修飾方法參數(shù)

參數(shù):遵循final修飾變量的約束條件,不能在方法中修改它的值或者指向別的對(duì)象。

 private void finalParam(final Map param){
    param = new HashMap();//報(bào)錯(cuò)
    param.put("","");//不報(bào)錯(cuò)
 }

1.3 final修飾方法

使用final方法的原因:確保在繼承中使方法行為保持不變,并且不會(huì)被覆蓋(設(shè)計(jì)考慮)。

  • final修飾的方法不可以重寫(xiě)(重寫(xiě)發(fā)生在父類(lèi)與之類(lèi))
  • final修飾的方法可以重載(同一個(gè)類(lèi))

以下代碼可以正確運(yùn)行:

public class FinalExampleParent {
    public final void test() {
    }
    public final void test(String str) {
    }
}

final和private:

類(lèi)中所有的private方法都隱式地指定為final的。由于其它類(lèi)無(wú)法取用private方法,因此無(wú)法覆蓋它。可以對(duì)private方法添加final修飾,但沒(méi)意義。

1.4 final修飾類(lèi)

當(dāng)類(lèi)定義為final時(shí),表示該類(lèi)不可繼承。
final類(lèi)的所有方法都是隱式為final,因?yàn)闊o(wú)法覆蓋它們

1.5 空白final

定義:被聲明為final但又未給定初值的域。
用途:提供了更大的靈活性:一個(gè)類(lèi)中的final域就可以做到根據(jù)對(duì)象而有所不同,卻又保持其恒定不變的特性。

class Poppet {
    private int i;
    Poppet(int ii) { i = ii; }
}
public class BlankFinal {
    private final int i = 0; // Initialized final
    private final int j; // Blank final
    private final Poppet p; // Blank final reference
    //空f(shuō)inal構(gòu)造器中初始化
    public BlankFinal() {
        j = 1; // Initialize blank final
        p = new Poppet(1); // Initialize blank final reference
    }
    public BlankFinal(int x) {
        j = x; // Initialize blank final
        p = new Poppet(x); // Initialize blank final reference
    }
    public static void main(String[] args) {
    //空f(shuō)inal域在不同情形下賦予不一樣的初值
        new BlankFinal();
        new BlankFinal(47);
    }
}

說(shuō)明:

  • 必須在域的定義處或者每個(gè)構(gòu)造器中對(duì)final賦值,這正是fnal域在使用前總是被初始化的原因所在。
  • 一個(gè)類(lèi)中的final域可以根據(jù)對(duì)象而有所不同,卻又保持其不變的特性。

1.6 static final

  • 同時(shí)是static final 的字段占據(jù)一段不能改變的存儲(chǔ)空間,它必須在定義的時(shí)候進(jìn)行賦值,否則編譯器將不予通過(guò)【即使在構(gòu)造函數(shù)中初始化也不行】。
  • static修飾的字段并不屬于一個(gè)對(duì)象,而是屬于這個(gè)類(lèi)的?!緦?duì)一個(gè)類(lèi)創(chuàng)建多個(gè)對(duì)象,其static final 修飾的變量其實(shí)是指向同一個(gè)值】

2 jvm角度理解final不可變性

一、Javac編譯器

final變量的不變性由Javac編譯時(shí)來(lái)保證:(只能在編譯期而不能在運(yùn)行期中檢查)

javac編譯時(shí),進(jìn)入數(shù)據(jù)及控制流分析階段時(shí),F(xiàn)low.flow()會(huì)涉及以下檢查:檢查final變量是否有多次賦值,空白final變量是否在構(gòu)造函數(shù)中進(jìn)行過(guò)初始化。

這里參考:javac final變量未賦值檢測(cè)講解

二、JVM類(lèi)加載

final類(lèi)的不可變性由jvm進(jìn)行類(lèi)加載的校驗(yàn)階段來(lái)保證:

JVM類(lèi)加載的校驗(yàn)階段中,對(duì)元數(shù)據(jù)驗(yàn)證時(shí),包含final語(yǔ)義校驗(yàn):
1. 這個(gè)類(lèi)的父類(lèi)是否繼承了不允許被繼承的類(lèi)(被final修飾的類(lèi))
2. 類(lèi)中的字段、方法是否與父類(lèi)產(chǎn)生矛盾(例如覆蓋了父類(lèi)的final字段,或者出現(xiàn)不符合規(guī)則的方法重載,例如方法參數(shù)都一致,但返回值類(lèi)型卻不同等)

3 final多線程下可見(jiàn)性

定義:被final修飾的字段在構(gòu)造器中一旦被初始化完成,并且構(gòu)造器沒(méi)有把“this”的引用傳遞出去,那么在其他線程中就能看見(jiàn)final字段的值。
如代碼所示,變量i與j都具備可見(jiàn)性,它們無(wú)須同步就能被其他線程正確訪問(wèn)。

public static final int i;
public final int j;
static {
    i = 0;
    // 省略后續(xù)動(dòng)作 
}
{
    // 選擇在構(gòu)造函數(shù)中初始化
    j = 0;
    // 省略后續(xù)動(dòng)作
}

解讀:
final字段如果聲明時(shí)賦值,因?yàn)橹荒苜x值一次,因此即便存在并發(fā),也能確保只有唯一值
如果在構(gòu)造函數(shù)中賦值,在無(wú)引用溢出下,構(gòu)造函數(shù)是線程安全的,因此final字段也是線程安全

4 final域重排序規(guī)則

這方面內(nèi)容待研究,或者參考:final域重排序規(guī)則

5 面試常見(jiàn)問(wèn)題

5.1 所有的final修飾的字段都是編譯期常量嗎?

不是
編譯期常量指的就是程序在編譯時(shí)就能確定這個(gè)常量的具體值
非編譯期常量就是程序在運(yùn)行時(shí)才能確定常量的值 (運(yùn)行時(shí)常量)

public class Test {
    //編譯期常量
    final int i = 1;
    final static int J = 1;
    //非編譯期常量
    Random r = new Random();
    final int k = r.nextInt();
}

k的值由隨機(jī)數(shù)對(duì)象決定,所以不是所有的final修飾的字段都是編譯期常量,只是k的值在被初始化后無(wú)法被更改。

5.2 final類(lèi)型的類(lèi)如何拓展?

設(shè)計(jì)模式中最重要的兩種關(guān)系,一種是繼承/實(shí)現(xiàn),另外一種是組合關(guān)系。所以當(dāng)遇到不能用繼承的,應(yīng)該考慮用組合:

class MyString{
    private String innerString;
    // ...init & other methods
    // 支持老的方法
    public int length(){
        return innerString.length(); // 通過(guò)innerString調(diào)用老的方法
    }
    // 添加新方法
    public String toMyString(){
        //...
    }
}

5.3 如何理解private所修飾的方法是隱式的final?

類(lèi)中所有的private方法都隱式地指定為final,因?yàn)槠渌?lèi)無(wú)法調(diào)用private方法,因此無(wú)法覆蓋它??梢詫?duì)private方法添加final修飾,但沒(méi)意義

參考書(shū)籍:《Thinking in Java》 《深入理解java虛擬機(jī)》

到此這篇關(guān)于Java關(guān)鍵詞final解讀的文章就介紹到這了,更多相關(guān)Java關(guān)鍵詞final解讀內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Ubuntu16.04 64位下JDK1.7的安裝教程

    Ubuntu16.04 64位下JDK1.7的安裝教程

    這篇文章主要為大家詳細(xì)介紹了Ubuntu16.04 64位下JDK1.7的安裝教程,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-09-09
  • Java使用jmeter進(jìn)行壓力測(cè)試

    Java使用jmeter進(jìn)行壓力測(cè)試

    本篇文章簡(jiǎn)單講一下使用jmeter進(jìn)行壓力測(cè)試。其壓測(cè)思想就是 通過(guò)創(chuàng)建指定數(shù)量的線程,同時(shí)請(qǐng)求指定接口,來(lái)模擬指定數(shù)量用戶同時(shí)進(jìn)行某個(gè)操作的場(chǎng)景,感興趣的小伙伴們可以參考一下
    2021-07-07
  • java與c#的語(yǔ)法區(qū)別詳細(xì)介紹

    java與c#的語(yǔ)法區(qū)別詳細(xì)介紹

    由C#轉(zhuǎn)入Java一段時(shí)間了,總結(jié)下個(gè)人認(rèn)為的Java同C#語(yǔ)法之間的不同之處,有不同意見(jiàn)之處還望各位海涵,剛學(xué)Java時(shí)覺(jué)得語(yǔ)法同C#大致是相同的(應(yīng)該說(shuō)C#同Java大致相同
    2012-11-11
  • SpringBoot自定義HttpMessageConverter操作

    SpringBoot自定義HttpMessageConverter操作

    這篇文章主要介紹了SpringBoot自定義HttpMessageConverter的操作,具有很好的參考價(jià)值,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-08-08
  • springcloud nacos的賦值均衡和動(dòng)態(tài)刷新

    springcloud nacos的賦值均衡和動(dòng)態(tài)刷新

    nacos是一個(gè)分布式的配置中心和注冊(cè)發(fā)現(xiàn)中心,這篇文章主要介紹了springcloud nacos的賦值均衡和動(dòng)態(tài)刷新,需要的朋友可以參考下
    2024-05-05
  • Java函數(shù)式編程(三):列表的轉(zhuǎn)化

    Java函數(shù)式編程(三):列表的轉(zhuǎn)化

    這篇文章主要介紹了Java函數(shù)式編程(二):列表的轉(zhuǎn)化,lambda表達(dá)式不僅能幫助我們遍歷集合,并且可以進(jìn)行集合的轉(zhuǎn)化,需要的朋友可以參考下
    2014-09-09
  • Java中g(shù)etResourceAsStream用法分析

    Java中g(shù)etResourceAsStream用法分析

    這篇文章主要介紹了Java中g(shù)etResourceAsStream用法,較為詳細(xì)的分析了getResourceAsStream的功能及用法,需要的朋友可以參考下
    2015-06-06
  • springboot整合 beatlsql的實(shí)例代碼

    springboot整合 beatlsql的實(shí)例代碼

    這篇文章主要介紹了springboot整合 beatlsql的實(shí)例代碼,BeetSql是一個(gè)全功能DAO工具,同時(shí)具有hibernate 優(yōu)點(diǎn) & Mybatis優(yōu)點(diǎn)功能,有興趣的可以了解一下
    2017-05-05
  • 關(guān)于泛型擦除問(wèn)題的解決--Mybatis查詢類(lèi)型轉(zhuǎn)換

    關(guān)于泛型擦除問(wèn)題的解決--Mybatis查詢類(lèi)型轉(zhuǎn)換

    這篇文章主要介紹了關(guān)于泛型擦除問(wèn)題的解決--Mybatis查詢類(lèi)型轉(zhuǎn)換方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • MyBatis 源碼分析 之SqlSession接口和Executor類(lèi)

    MyBatis 源碼分析 之SqlSession接口和Executor類(lèi)

    mybatis框架在操作數(shù)據(jù)的時(shí)候,離不開(kāi)SqlSession接口實(shí)例類(lèi)的作用,下面通過(guò)本文給大家實(shí)例剖析MyBatis 源碼分析之SqlSession接口和Executor類(lèi),需要的朋友參考下吧
    2017-02-02

最新評(píng)論