成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

加個Final就能防止被修改?是我太naive了

開發 前端
要想回答上面的問題,我們首先得知道什么是不變性(Immutable)。如果對象在被創建之后,其狀態就不能修改了,那么它就具備“不變性”。
本文轉載自微信公眾號「JerryCodes」,作者KyleJerry 。轉載本文請聯系JerryCodes公眾號。
  • 什么是不變性
  • final 和不可變的關系
  • 總結

什么是不變性

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

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

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

如果我們創建一個 person 對象,那么里面的屬性會有兩個,即 id 和 age,并且由于它們都是被 final 修飾的,所以一旦這個 person 對象被創建好,那么它里面所有的屬性,即 id 和 age 就都是不能變的。我們如果想改變其中屬性的值就會報錯,代碼如下所示:

  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 對象就具備不變性,也就意味著它的狀態是不能改變的。

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

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

被 final 修飾的變量意味著一旦被賦值就不能修改,也就是只能被賦值一次,如果我們嘗試對已經被 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的變量(數組) 
  15.     } 

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

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

  1. class Test { 
  2.     public static void main(String args[]) { 
  3.        final int arr[] = {1, 2, 3, 4, 5};  //  注意,數組 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.     } 

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

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

但是我們現在想證明的是,數組對象里面的內容可以修改,所以接下來我們就用 for 循環把它里面的內容都乘以 10,最后打印出來結果如下:

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

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

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

  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 函數中新建了 Test 的實例 t 之后,把它用 final 修飾,然后去嘗試改它里面成員變量 p 的值,并打印出結果,程序會打印出“30”。一開始 p 的值是 20,但是最后修改完畢變成了 30,說明這次修改是成功的。

以上我們就得出了一個結論,final 修飾一個指向對象的變量的時候,對象本身的內容依然是可以變化的。

final 和不可變的關系

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

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

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

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

  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 修飾的成員變量,并且這個成員變量不是基本類型,而是對象類型,那么情況就不一樣了。有了前面基礎之后,我們知道,對于對象類型的屬性而言,我們如果給它加了 final,它內部的成員變量還是可以變化的,因為 final 只能保證其引用不變,不能保證其內容不變。所以這個時候若一旦某個對象類型的內容發生了變化,就意味著這整個類都不具備不變性了。

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

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

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

代碼如下:

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

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

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

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

總結

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

 

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

 

責任編輯:武曉燕 來源: JerryCodes
相關推薦

2011-03-08 09:41:49

2017-05-24 17:19:55

CBP設備加密

2022-06-08 10:01:06

Go語法PHP

2021-07-21 09:00:00

面部識別AI安全

2024-02-19 00:00:00

接口圖形驗證碼

2011-06-09 13:26:27

2011-06-09 12:50:47

2018-04-27 10:33:56

Linux命令chattr

2013-06-20 11:11:00

程序員經理

2024-06-18 08:31:33

2015-05-05 14:50:21

Python不需要操作系統

2024-06-12 12:13:48

2022-08-06 13:04:27

LinuxSHH

2012-12-27 10:05:15

云計算校園一卡通云管理平臺

2018-01-08 09:52:23

CEO技術合伙人

2021-01-27 17:24:27

密碼root權限漏洞

2024-01-23 17:33:36

2013-01-10 15:32:58

App StoreiOS虛假應用

2024-01-15 09:15:52

parallel語句函數

2020-12-18 08:28:13

Redis數據數據庫
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 91色视频在线观看 | 久久久久国产一区二区三区四区 | 欧美中文字幕一区二区三区亚洲 | 久久国产精品无码网站 | 亚洲国产精品一区在线观看 | 久久偷人 | 国产精品欧美一区二区 | 亚洲一区免费 | 欧美国产日韩精品 | 国产精品久久av | 成人免费淫片aa视频免费 | 久久久国产一区二区三区 | 国产精品一区久久久 | 国产乱码精品一区二区三区忘忧草 | 欧美一区成人 | 正在播放国产精品 | chinese中国真实乱对白 | 国产精品国产成人国产三级 | 亚洲欧美在线观看 | 成人免费影院 | 在线一级片 | 欧美精品在线免费观看 | 毛片链接 | 九九久视频 | 国产精品视频免费观看 | 日韩中文字幕免费在线观看 | 极品的亚洲 | 午夜影院在线播放 | 国产精品精品视频 | 青草青草久热精品视频在线观看 | 成人免费精品 | 久久亚洲视频 | 激情欧美一区二区三区 | 亚州综合一区 | 久久精品视频一区二区三区 | 天天影视综合 | 亚洲精品一二区 | 色免费看 | 一区二区三区中文 | 天天操天天插 | 久久久久久久一区二区三区 |