深入理解java final不可變性
一、java final基本概念:
1、主要用于修飾類(lèi)、屬性和方法:
被final修飾的類(lèi)不可以被繼承
被final修飾的方法不可以被重寫(xiě)
被final修飾的變量不可以被改變,被final修飾不可變的是變量的引用,而不是引用指向的內(nèi)容,引用指向的內(nèi)容是可以改變的
2、final finally finalize區(qū)別:
(1)final:可以修飾類(lèi)、變量、方法,修飾類(lèi)表示該類(lèi)不能被繼承、修飾方法表示該方法不能被重寫(xiě)、修飾變量表
示該變量是一個(gè)常量不能被重新賦值。
(2)finally:一般作用在try-catch代碼塊中,在處理異常的時(shí)候,通常我們將一定要執(zhí)行的代碼方法finally代碼塊
中,表示不管是否出現(xiàn)異常,該代碼塊都會(huì)執(zhí)行,一般用來(lái)存放一些關(guān)閉資源的代碼。
(3)finalize:是一個(gè)方法,屬于Object類(lèi)的一個(gè)方法,而Object類(lèi)是所有類(lèi)的父類(lèi),該方法一般由垃圾回收器來(lái)調(diào)
用,當(dāng)我們調(diào)用System.gc() 方法的時(shí)候,由垃圾回收器調(diào)用finalize(),回收垃圾,一個(gè)對(duì)象是否可回收的
二、final 不可變:
1、JAVA String類(lèi) 為什么是final 不可變的?
(1)為了線程安全:
(2)為了實(shí)現(xiàn)字符串池:
字符串池: java中的字符串池是存儲(chǔ)在Java堆內(nèi)存中的字符串池;
簡(jiǎn)單來(lái)說(shuō),字符串池就是相當(dāng)于一個(gè)公用相冊(cè),需要的時(shí)候從公用相冊(cè)里面去查找,如果有就直接使用,如果沒(méi)有就添加一張進(jìn)去,這樣自己和其他人就可以使用了
(3)為了實(shí)現(xiàn)String可以創(chuàng)建HashCode不可變性:
先看下string類(lèi)的源碼,被final修飾,final修飾的char[]代表了被存儲(chǔ)的數(shù)據(jù)不可更改性,源碼如下:
a. String為什么要被final修飾:
主要是為了安全性和效率;
b. final修飾的String:
代表了String的不可繼承性,final修飾的char[]代表了被存儲(chǔ)的數(shù)據(jù)不可更改性。但是:雖然final代表了不可變,但僅僅是引用地址不可變,并不代表了數(shù)組本身不會(huì)變
c. final修改例子分析:
在例1中,我們用final修飾了一個(gè)集合list,并對(duì)集合進(jìn)行add()操作,執(zhí)行成功。
在例2中,我們對(duì)集合進(jìn)行變更,執(zhí)行失敗;
d. 原因分析:
final修飾的集合‘list'是一個(gè)引用,而這個(gè)引用指向了‘list',在往集合里添加數(shù)據(jù)的時(shí)候,并沒(méi)有影響到‘stringList'引用地址。而當(dāng)我們 list = new ArrayList<>(); 為什么就不可以了呢? 因?yàn)檫@就相當(dāng)于修改引用地址,是不可以的。final的意思是地址不能改,但是地址指向的內(nèi)容當(dāng)然可以改
數(shù)組是私有方法,所以起作用的還有private,正是因?yàn)閮烧弑WC了String的不可變性。
e. 只有當(dāng)字符串是不可變的,字符串池才有可能實(shí)現(xiàn):
一個(gè)緩存區(qū),如果字符串可變,如果在其它地方也有引用,指向的引用發(fā)生了變更,會(huì)發(fā)生混亂;
字符串池的實(shí)現(xiàn)可以在運(yùn)行時(shí)節(jié)約很多heap空間,因?yàn)椴煌淖址兞慷贾赶虺刂械耐粋€(gè)字符串。但如果字符串是可變的,那么String interning將不能實(shí)現(xiàn),因?yàn)檫@樣的話,如果變量改變了它的值,那么其它指向這個(gè)值的變量的值也會(huì)一起改變
f. hashcode不可變性
因?yàn)樽址遣豢勺兊?,所以在它?chuàng)建的時(shí)候HashCode就被緩存了,不需要重新計(jì)算。這就使得字符串很適合作為Map中的鍵,字符串的處理速度要快過(guò)其它的鍵對(duì)象。這就是HashMap中的鍵往往都使用字符串
得出來(lái)的HashCode是一樣的,但是為什么得出來(lái)的地址不一樣
因?yàn)橹苯佣x的String m = “a”; 是儲(chǔ)存在常量存儲(chǔ)區(qū)中的字符串常量池中;而new String(“a”)是存儲(chǔ)在堆中,所以地址不一樣
2、String的不可變性:
(1)為什么需要保證String不可變
因?yàn)橹挥挟?dāng)字符串是不可變的,字符串池才有可能實(shí)現(xiàn)。字符串池的實(shí)現(xiàn)可以在運(yùn)行時(shí)節(jié)約很多heap空間,因?yàn)椴煌淖址兞慷贾赶虺刂械耐粋€(gè)字符串。但如果字符串是可變的,那么String interning將不能實(shí)現(xiàn),因?yàn)檫@樣的話,如果變量改變了它的值,那么其它指向這個(gè)值的變量的值也會(huì)一起改變。
(2)如果字符串是可變的,那么會(huì)引起很?chē)?yán)重的安全問(wèn)題
比如,數(shù)據(jù)庫(kù)的用戶名、密碼都是以字符串的形式傳入來(lái)獲得數(shù)據(jù)庫(kù)的連接,或者在socket編程中,主機(jī)名和端口都是以字符串的形式傳入。因?yàn)樽址遣豢勺兊?,所以它的值是不可改變的,否則黑客們可以鉆到空子,改變字符串指向的對(duì)象的值,造成安全漏洞。
(3)因?yàn)樽址遣豢勺兊模允嵌嗑€程安全的
同一個(gè)字符串實(shí)例可以被多個(gè)線程共享,這樣便不用因?yàn)榫€程安全問(wèn)題而使用同步,字符串自己便是線程安全的。
(4)因?yàn)樽址遣豢勺兊?,所以在它?chuàng)建的時(shí)候HashCode就被緩存了,不需要重新計(jì)算。
這就使得字符串很適合作為Map中的鍵,字符串的處理速度要快過(guò)其它的鍵對(duì)象,這就是HashMap中的鍵往往都使用字符串
到此這篇關(guān)于講講java final不可變性的文章就介紹到這了,更多相關(guān)java final不可變性 內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring?Data?JPA框架的Repository自定義實(shí)現(xiàn)詳解
Spring?Data?JPA是Spring基于JPA規(guī)范的基礎(chǔ)上封裝的?套?JPA?應(yīng)?框架,可使開(kāi)發(fā)者?極簡(jiǎn)的代碼即可實(shí)現(xiàn)對(duì)數(shù)據(jù)庫(kù)的訪問(wèn)和操作,本篇我們來(lái)了解Spring?Data?JPA框架的Repository自定義實(shí)現(xiàn)2022-04-04Spring整合多數(shù)據(jù)源實(shí)現(xiàn)動(dòng)態(tài)切換的實(shí)例講解
下面小編就為大家?guī)?lái)一篇Spring整合多數(shù)據(jù)源實(shí)現(xiàn)動(dòng)態(tài)切換的實(shí)例講解。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-07-07java基于JDBC連接Oracle 11g Release2實(shí)例分析
這篇文章主要介紹了java基于JDBC連接Oracle 11g Release2的方法,實(shí)例分析了JDBC連接Oracle 11g Release2容易出現(xiàn)的異常與解決方法,需要的朋友可以參考下2015-06-06手?jǐn)]一個(gè) spring-boot-starter的全過(guò)程
這篇文章主要介紹了手?jǐn)]一個(gè) spring-boot-starter的全過(guò)程,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-01-01