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

Java 中如何實現一個像 String 一樣不可變的類?

開發 前端
我們思考一下,上面的代碼是否真正的做到了不可變,好,我們思考三秒鐘,心里默默的數三下。

如果問你在日常開發中用到的最多的一個 Java? 類是什么,阿粉敢打賭絕對是 String.class?。說到 String? 大家都知道 String 是一個不可變的類;雖然用的很多,那不知道小伙伴們有沒有想過怎么樣創建一個自己的不可變的類呢?這篇文章阿粉就帶大家來實踐一下,創建一個自己的不可變的類。

特性

在手動編寫代碼之前,我們先了解一下不可變類都有哪些特性,

定義類的時候需要使用final? 關鍵字進行修飾:之所以使用 final 進行修飾是因為這樣可以避免被其他類繼承,一旦有了子類繼承就會破壞父類的不可變性機制;

成員變量需要使用fina?l 關鍵詞修飾,并且需要是 private 的:避免屬性被外部修改;

成員變量不可提供setter? 方法,只能提供 getter 方法:避免被外部修改,并且避免返回成員變量本身;

提供所有字段的構造函數;

實操

知道了不可變類的一些基本特性之后,我們來實際寫代碼操作一下,以及我們會驗證一下,如果不按照上面的要求來編寫的話,會出現什么樣的問題。

這里我們定義一個 Teacher? 類來測試一下,按照我們上面提到的幾點,我們給類和屬性的定義都加上 final 代碼如下所示。

package com.example.demo.immutable;

import java.util.List;
import java.util.Map;

public final class Teacher {
private final String name;
private final List<String> students;
private final Address address;
private final Map<String, String> metadata;

public Teacher(String name, List<String> students, Address address, Map<String, String> metadata){
this.name = name;
this.students = students;
this.address = address;
this.metadata = metadata;
}

public String getName(){
return name;
}


public List<String> getStudents(){
return students;
}

public Address getAddress(){
return address;
}

public Map<String, String> getMetadata(){
return metadata;
}
}
package com.example.demo.immutable;

public class Address {
private String country;
private String city;

public String getCountry(){
return country;
}

public void setCountry(String country){
this.country = country;
}

public String getCity(){
return city;
}

public void setCity(String city){
this.city = city;
}
}

我們思考一下,上面的代碼是否真正的做到了不可變,好,我們思考三秒鐘,心里默默的數三下。為了回答這個問題,我們看下下面的測試代碼。

package com.example.demo;

import com.example.demo.immutable.Address;
import com.example.demo.immutable.Teacher;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

/**
* <br>
* <b>Function:</b><br>
* <b>Author:</b>@author Silence<br>
* <b>Date:</b>2022-11-22 21:17<br>
* <b>Desc:</b>無<br>
*/
public class ImmutableDemo {

public static void main(String[] args){
List<String> students = new ArrayList<>();
students.add("鴨血粉絲 1");
students.add("鴨血粉絲 2");
students.add("鴨血粉絲 3");
Address address = new Address();
address.setCountry("中國");
address.setCity("深圳");
Map<String, String> metadata = new HashMap<>();
metadata.put("hobby", "籃球");
metadata.put("age", "29");
Teacher teacher = new Teacher("Java極客技術", students, address, metadata);
System.out.println(teacher.getStudents().size());
System.out.println(teacher.getMetadata().size());
System.out.println(teacher.getAddress().getCity());

// 修改屬性
teacher.getStudents().add("小明");
teacher.getMetadata().put("weight", "120");
teacher.getAddress().setCity("廣州");

System.out.println(teacher.getStudents().size());
System.out.println(teacher.getMetadata().size());
System.out.println(teacher.getAddress().getCity());
}

}

運行的結果如下截圖所示,通過測試我們可以發現,簡單的只添加 final? 關鍵字是不能解決不可變性的,我們當前的 teacher 實例已經被外層修改掉了成員變量。

圖片

為了解決這個問題,我們還需要對我們的 Teacher? 類進行改造,首先我們可以想到的就是需要將 students? 和 metadata? 兩個成員變量不能直接返回給外層,否則外層的修改會直接影響到我們的不可變類,那么我們就可以修改 getter 方法,拷貝一下成員變量進行返回,而不是直接返回,修改代碼如下

public List<String> getStudents(){
return new ArrayList<>(students);
//return students;
}
public Map<String, String> getMetadata(){
return new HashMap<>(metadata);
//return metadata;
}

我們再次運行上面的測試代碼,可以看到這次的返回數據如下,這次我們的 students? 和 metadate? 成員變量并沒有被外層修改掉了。但是我們的 address 成員變量還是有問題,沒關系,我們接著往下看。

圖片

很自然的為了解決 address? 的問題,我們想到了也是進行一個拷貝,再調用 getter? 方法的時候返回一個拷貝對象,而不是直接返回成員變量。那我們就需要改造 Address? 類,將其變成 Cloneable? 的即可,我們實現 接口,然后覆蓋一個 clone 方法,代碼如下

package com.example.demo.immutable;

public class Address implements Cloneable{
...// 省略
@Override
public Address clone(){
try {
return (Address) super.clone();
} catch (CloneNotSupportedException e) {
throw new AssertionError();
}
}
}

再修改 Teacher? 的 getAddress 方法

public Address getAddress(){
//return address;
return address.clone();
}

接下來我們再運行一下測試代碼,結果如下,可以看到這次我們的 teacher 實例的成員變量并沒有被修改掉了,至此我們完成了一個不可變對象的創建!

圖片

String 的實現

前面我們看的是自定義實現不可變類的操作,接下來我們簡單看一下 String? 類是如何實現不可變的,通過源碼我們可以看到 String? 也使用了關鍵字 final? 來避免被子類繼承,以及對應存放具體值的成員變量也使用了 final 關鍵字。

圖片

并且對外提供的方法 substring? 也是通過復制的形式對外提供的新的 String 對象。

圖片

圖片

注意阿粉這里的 JDK 版本是 19 所以可能大家版本不一致具體的實現不太一樣,但是本質上都是一樣的。

責任編輯:武曉燕 來源: Java極客技術
相關推薦

2023-09-04 14:28:33

FlarumDiscourse開源

2015-02-11 10:42:48

Apple Watch

2013-08-22 10:17:51

Google大數據業務價值

2015-03-16 12:50:44

2020-12-07 10:10:22

企業文化客戶體驗

2023-04-05 14:19:07

FlinkRedisNoSQL

2016-10-08 00:24:53

httptcp實時

2017-05-22 10:33:14

PythonJuliaCython

2021-09-07 10:29:11

JavaScript模塊CSS

2023-05-23 13:59:41

RustPython程序

2013-12-31 09:19:23

Python調試

2013-12-17 09:02:03

Python調試

2022-12-21 15:56:23

代碼文檔工具

2020-10-11 21:39:35

計算機互聯網 技術

2023-02-15 08:17:20

VSCodeTypeScrip

2021-11-04 17:23:03

Java對象 immutable

2013-07-26 10:15:29

云計算大數據Hadoop

2015-10-12 08:56:27

Java不可變

2017-11-06 14:18:03

2020-08-25 08:56:55

Pythonawk字符串
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲国产免费 | 欧美精品电影一区 | 波多野结衣一区二区三区 | 成人一区在线观看 | 一区二区三区韩国 | 一二三区av| 欧美区在线 | 久久久久久久久蜜桃 | 欧美成人aaa级毛片在线视频 | 欧美一区二区三区久久精品视 | 色婷婷亚洲国产女人的天堂 | 久久久久久中文字幕 | 欧美激情国产日韩精品一区18 | 精品一区二区三区四区视频 | 欧美日韩亚洲系列 | 天天拍天天操 | 色狠狠一区 | 国产一区不卡在线观看 | 亚洲视频一区在线 | 天天躁日日躁狠狠躁2018小说 | 国产成人免费视频网站高清观看视频 | 国产精品99久久久久久久vr | 国产成人精品一区二区三区 | 欧美国产视频一区二区 | 97偷拍视频 | 99久久夜色精品国产亚洲96 | 国产在线激情视频 | www.亚洲.com| 久久99精品久久久久久国产越南 | 男女免费在线观看视频 | 亚洲成人精品 | 亚洲一区二区在线免费观看 | 国产在线精品一区二区三区 | 国产精品久久久久久久三级 | 天天操夜夜骑 | 国产亚洲人成a在线v网站 | 精品福利av导航 | 暖暖日本在线视频 | 亚洲精品一区二三区不卡 | 国产不卡在线观看 | 欧美a在线看 |