偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

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

開發(fā) 前端
要想回答上面的問題,我們首先得知道什么是不變性(Immutable)。如果對(duì)象在被創(chuàng)建之后,其狀態(tài)就不能修改了,那么它就具備“不變性”。
本文轉(zhuǎn)載自微信公眾號(hào)「JerryCodes」,作者KyleJerry 。轉(zhuǎn)載本文請(qǐng)聯(lián)系JerryCodes公眾號(hào)。
  • 什么是不變性
  • final 和不可變的關(guān)系
  • 總結(jié)

什么是不變性

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

我們舉個(gè)例子,比如下面這個(gè) Person 類:

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

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

  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;//編譯錯(cuò)誤,無法修改 final 變量的值 
  9.     } 

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

final 修飾對(duì)象時(shí),只是引用不可變!

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

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

  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;     //編譯錯(cuò)誤,不允許修改final的變量(基本類型) 
  13. //        finalVarCantChange.random=null;    //編譯錯(cuò)誤,不允許修改final的變量(對(duì)象) 
  14. //        finalVarCantChange.array = new int[5];//編譯錯(cuò)誤,不允許修改final的變量(數(shù)組) 
  15.     } 

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

這就證明了“被 final 修飾的變量意味著一旦被賦值就不能修改”,而這個(gè)規(guī)則對(duì)于基本類型的變量是沒有歧義的,但是對(duì)于對(duì)象類型而言,final 其實(shí)只是保證這個(gè)變量的引用不可變,而對(duì)象本身依然是可以變化的。這一點(diǎn)同樣適用于數(shù)組,因?yàn)樵?Java 中數(shù)組也是對(duì)象。那我們就來舉個(gè)例子,看一看以下 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.     } 

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

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

但是我們現(xiàn)在想證明的是,數(shù)組對(duì)象里面的內(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)容依然是可以被修改的。

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

  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.     } 

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

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

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

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

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

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

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

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

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

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

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

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

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

代碼如下:

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

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

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

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

總結(jié)

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

 

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

 

責(zé)任編輯:武曉燕 來源: JerryCodes
相關(guān)推薦

2011-03-08 09:41:49

2017-05-24 17:19:55

CBP設(shè)備加密

2021-07-21 09:00:00

面部識(shí)別AI安全

2024-02-19 00:00:00

接口圖形驗(yàn)證碼

2022-06-08 10:01:06

Go語法PHP

2013-06-20 11:11:00

程序員經(jīng)理

2011-06-09 13:26:27

2011-06-09 12:50:47

2024-06-18 08:31:33

2015-05-05 14:50:21

Python不需要操作系統(tǒng)

2018-04-27 10:33:56

Linux命令chattr

2012-12-27 10:05:15

云計(jì)算校園一卡通云管理平臺(tái)

2024-06-12 12:13:48

2022-08-06 13:04:27

LinuxSHH

2021-01-27 17:24:27

密碼root權(quán)限漏洞

2018-01-08 09:52:23

CEO技術(shù)合伙人

2024-01-23 17:33:36

2024-01-15 09:15:52

parallel語句函數(shù)

2020-12-18 08:28:13

Redis數(shù)據(jù)數(shù)據(jù)庫

2020-05-14 14:54:00

GitHub星級(jí)開源
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)