日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
加個Final就能防止被修改?是我太naive了

本文轉(zhuǎn)載自微信公眾號「JerryCodes」,作者KyleJerry 。轉(zhuǎn)載本文請聯(lián)系JerryCodes公眾號。

創(chuàng)新互聯(lián)堅持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:網(wǎng)站設(shè)計制作、做網(wǎng)站、企業(yè)官網(wǎng)、英文網(wǎng)站、手機端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿足客戶于互聯(lián)網(wǎng)時代的和縣網(wǎng)站設(shè)計、移動媒體設(shè)計的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!

  • 什么是不變性
  • final 和不可變的關(guān)系
  • 總結(jié)

什么是不變性

要想回答上面的問題,我們首先得知道什么是不變性(Immutable)。如果對象在被創(chuàng)建之后,其狀態(tài)就不能修改了,那么它就具備“不變性”。

我們舉個例子,比如下面這個 Person 類:

 
 
 
  1. public class Person { 
  2.  
  3.     final int id = 1; 
  4.     final int age = 18; 

如果我們創(chuàng)建一個 person 對象,那么里面的屬性會有兩個,即 id 和 age,并且由于它們都是被 final 修飾的,所以一旦這個 person 對象被創(chuàng)建好,那么它里面所有的屬性,即 id 和 age 就都是不能變的。我們?nèi)绻敫淖兤渲袑傩缘闹稻蜁箦e,代碼如下所示:

 
 
 
  1. public class Person { 
  2.  
  3.     final int id = 1; 
  4.     final int age = 18; 
  5.  
  6.     public static void main(String[] args) { 
  7.         Person person = new Person(); 
  8. //        person.age=5;//編譯錯誤,無法修改 final 變量的值 
  9.     } 

比如我們嘗試去改變這個 person 對象,例如將 age 改成 5,則會編譯通不過,所以像這樣的 person 對象就具備不變性,也就意味著它的狀態(tài)是不能改變的。

final 修飾對象時,只是引用不可變!

這里有個非常重要的注意點,那就是當我們用 final 去修飾一個指向?qū)ο箢愋?而不是指向 8 種基本數(shù)據(jù)類型,例如 int 等)的變量時候,那么 final 起到的作用只是保證這個變量的引用不可變,而對象本身的內(nèi)容依然是可以變化的。下面我們對此展開講解。

被 final 修飾的變量意味著一旦被賦值就不能修改,也就是只能被賦值一次,如果我們嘗試對已經(jīng)被 final 修飾過的變量再次賦值的話,則會報編譯錯誤。我們用下面的代碼來說明:

 
 
 
  1. /** 
  2.  * 描述:     final變量一旦被賦值就不能被修改 
  3.  */ 
  4. public class FinalVarCantChange { 
  5.  
  6.     private final int finalVar = 0; 
  7.     private final Random random = new Random(); 
  8.     private final int array[] = {1,2,3}; 
  9.  
  10.     public static void main(String[] args) { 
  11.         FinalVarCantChange finalVarCantChange = new FinalVarCantChange(); 
  12. //        finalVarCantChange.finalVar=9;     //編譯錯誤,不允許修改final的變量(基本類型) 
  13. //        finalVarCantChange.random=null;    //編譯錯誤,不允許修改final的變量(對象) 
  14. //        finalVarCantChange.array = new int[5];//編譯錯誤,不允許修改final的變量(數(shù)組) 
  15.     } 

我們首先在這里分別創(chuàng)建了一個 int 類型的變量、一個 Random 類型的變量,還有一個是數(shù)組,它們都是被 final 修飾的;然后嘗試對它們進行修改,比如把 int 變量的值改成 9,或者把 random 變量置為 null,或者給數(shù)組重新指定一個內(nèi)容,這些代碼都無法通過編譯。

這就證明了“被 final 修飾的變量意味著一旦被賦值就不能修改”,而這個規(guī)則對于基本類型的變量是沒有歧義的,但是對于對象類型而言,final 其實只是保證這個變量的引用不可變,而對象本身依然是可以變化的。這一點同樣適用于數(shù)組,因為在 Java 中數(shù)組也是對象。那我們就來舉個例子,看一看以下 Java 程序的輸出:

 
 
 
  1. class Test { 
  2.     public static void main(String args[]) { 
  3.        final int arr[] = {1, 2, 3, 4, 5};  //  注意,數(shù)組 arr 是 final 的 
  4.        for (int i = 0; i < arr.length; i++) { 
  5.            arr[i] = arr[i]*10; 
  6.            System.out.println(arr[i]); 
  7.        } 
  8.     } 

首先來猜測一下,假設(shè)不看下面的輸出結(jié)果,只看這段代碼,你猜它打印出什么樣的結(jié)果?

這段代碼中有個 Test 類,而且這個類只有一個 main 方法,方法里面有一個 final 修飾的 arr 數(shù)組。注意,數(shù)組是對象的一種,現(xiàn)在數(shù)組是被 final 修飾的,所以它的意思是一旦被賦值之后,變量的引用不能修改。

但是我們現(xiàn)在想證明的是,數(shù)組對象里面的內(nèi)容可以修改,所以接下來我們就用 for 循環(huán)把它里面的內(nèi)容都乘以 10,最后打印出來結(jié)果如下:

 
 
 
  1. 10  
  2. 20  
  3. 30  
  4. 40  
  5. 50 

可以看到,它打印出來的是 10 20 30 40 50,而不是最開始的 1 2 3 4 5,這就證明了,雖然數(shù)組 arr 被 final 修飾了,它的引用不能被修改,但是里面的內(nèi)容依然是可以被修改的。

同樣,對于非數(shù)組的對象而言也是如此,我們來看下面的例子:

 
 
 
  1. class Test {  
  2.     int p = 20;  
  3.     public static void main(String args[]){  
  4.        final Test t = new Test(); 
  5.        t.p = 30;  
  6.        System.out.println(t.p); 
  7.     } 

這個 Test 類中有一個 int 類型的 p 屬性,我們在 main 函數(shù)中新建了 Test 的實例 t 之后,把它用 final 修飾,然后去嘗試改它里面成員變量 p 的值,并打印出結(jié)果,程序會打印出“30”。一開始 p 的值是 20,但是最后修改完畢變成了 30,說明這次修改是成功的。

以上我們就得出了一個結(jié)論,final 修飾一個指向?qū)ο蟮淖兞康臅r候,對象本身的內(nèi)容依然是可以變化的。

final 和不可變的關(guān)系

這里就引申出一個問題,那就是 final 和不變性究竟是什么關(guān)系?

那我們就來具體對比一下 final 和不變性。關(guān)鍵字 final 可以確保變量的引用保持不變,但是不變性意味著對象一旦創(chuàng)建完畢就不能改變其狀態(tài),它強調(diào)的是對象內(nèi)容本身,而不是引用,所以 final 和不變性這兩者是很不一樣的。

對于一個類的對象而言,你必須要保證它創(chuàng)建之后所有內(nèi)部狀態(tài)(包括它的成員變量的內(nèi)部屬性等)永遠不變,才是具有不變性的,這就要求所有成員變量的狀態(tài)都不允許發(fā)生變化。

有一種說法就認為:“要想保證對象具有不變性的最簡單的辦法,就是把類中所有屬性都聲明為 final”,這條規(guī)則是不完全正確的,它通常只適用于類的所有屬性都是基本類型的情況,比如前面的例子:

 
 
 
  1. public class Person { 
  2.  
  3.     final int id = 1; 
  4.     final int age = 18; 

Person 類里面有 final int id 和 final int age 兩個屬性,都是基本類型的,且都加了 final,所以 Person 類的對象確實是具備不變性的。

但是如果一個類里面有一個 final 修飾的成員變量,并且這個成員變量不是基本類型,而是對象類型,那么情況就不一樣了。有了前面基礎(chǔ)之后,我們知道,對于對象類型的屬性而言,我們?nèi)绻o它加了 final,它內(nèi)部的成員變量還是可以變化的,因為 final 只能保證其引用不變,不能保證其內(nèi)容不變。所以這個時候若一旦某個對象類型的內(nèi)容發(fā)生了變化,就意味著這整個類都不具備不變性了。

所以我們就得出了這個結(jié)論:不變性并不意味著,簡單地使用 final 修飾所有類的屬性,這個類的對象就具備不變性了。

那就會有一個很大的疑問,假設(shè)我的類里面有一個對象類型的成員變量,那要怎樣做才能保證整個對象是不可變的呢?

我們來舉個例子,即一個包含對象類型的成員變量的類的對象,具備不可變性的例子。

代碼如下:

 
 
 
  1. public class ImmutableDemo { 
  2.  
  3.     private final Set lessons = new HashSet<>(); 
  4.  
  5.     public ImmutableDemo() { 
  6.         lessons.add("第01講:為何說只有 1 種實現(xiàn)線程的方法?"); 
  7.         lessons.add("第02講:如何正確停止線程?為什么 volatile 標記位的停止方法是錯誤的?"); 
  8.         lessons.add("第03講:線程是如何在 6 種狀態(tài)之間轉(zhuǎn)換的?"); 
  9.     } 
  10.  
  11.     public boolean isLesson(String name) { 
  12.         return lessons.contains(name); 
  13.     } 

在這個類中有一個 final 修飾的、且也是 private 修飾的的一個 Set 對象,叫作 lessons,它是個 HashSet;然后我們在構(gòu)造函數(shù)中往這個 HashSet 里面加了三個值,分別是第 01、02、03 講的題目;類中還有一個方法,即 isLesson,去判斷傳入的參數(shù)是不是屬于本課前 3 講的標題,isLesson 方法就是利用 lessons.contains 方法去判斷的,如果包含就返回 true,否則返回 false。這個類的內(nèi)容就是這些了,沒有其他額外的代碼了。

在這種情況下,盡管 lessons 是 Set 類型的,盡管它是一個對象,但是對于 ImmutableDemo 類的對象而言,就是具備不變性的。因為 lessons 對象是 final 且 private 的,所以引用不會變,且外部也無法訪問它,而且 ImmutableDemo 類也沒有任何方法可以去修改 lessons 里包含的內(nèi)容,只是在構(gòu)造函數(shù)中對 lessons 添加了初始值,所以 ImmutableDemo 對象一旦創(chuàng)建完成,也就是一旦執(zhí)行完構(gòu)造方法,后面就再沒有任何機會可以修改 lessons 里面的數(shù)據(jù)了。

而對于 ImmutableDemo 類而言,它就只有這么一個成員變量,而這個成員變量一旦構(gòu)造完畢之后又不能變,所以就使得這個 ImmutableDemo 類的對象是具備不變性的,這就是一個很好的“包含對象類型的成員變量的類的對象,具備不可變性”的例子。

總結(jié)

我們首先介紹了什么是不變性,然后介紹了用 final 修飾一個對象類型的變量的時候,只能保證它的引用不變,但是對象內(nèi)容自身依然是可以變的。

僅僅把所有的成員變量都用 final 修飾并不能代表類的對象就是具備不變性的。


網(wǎng)頁標題:加個Final就能防止被修改?是我太naive了
網(wǎng)站URL:http://www.5511xx.com/article/cohshpp.html