自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Java 中如何實(shí)現(xiàn)一個像 String 一樣不可變的類?

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

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

特性

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

定義類的時候需要使用final? 關(guān)鍵字進(jìn)行修飾:之所以使用 final 進(jìn)行修飾是因?yàn)檫@樣可以避免被其他類繼承,一旦有了子類繼承就會破壞父類的不可變性機(jī)制;

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

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

提供所有字段的構(gòu)造函數(shù);

實(shí)操

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

這里我們定義一個 Teacher? 類來測試一下,按照我們上面提到的幾點(diǎn),我們給類和屬性的定義都加上 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;
}
}

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

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極客技術(shù)", 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());
}

}

運(yùn)行的結(jié)果如下截圖所示,通過測試我們可以發(fā)現(xiàn),簡單的只添加 final? 關(guān)鍵字是不能解決不可變性的,我們當(dāng)前的 teacher 實(shí)例已經(jīng)被外層修改掉了成員變量。

圖片

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

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

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

圖片

很自然的為了解決 address? 的問題,我們想到了也是進(jìn)行一個拷貝,再調(diào)用 getter? 方法的時候返回一個拷貝對象,而不是直接返回成員變量。那我們就需要改造 Address? 類,將其變成 Cloneable? 的即可,我們實(shí)現(xiàn) 接口,然后覆蓋一個 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();
}

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

圖片

String 的實(shí)現(xiàn)

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

圖片

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

圖片

圖片

注意阿粉這里的 JDK 版本是 19 所以可能大家版本不一致具體的實(shí)現(xiàn)不太一樣,但是本質(zhì)上都是一樣的。

責(zé)任編輯:武曉燕 來源: Java極客技術(shù)
相關(guān)推薦

2023-09-04 14:28:33

FlarumDiscourse開源

2015-03-16 12:50:44

2013-08-22 10:17:51

Google大數(shù)據(jù)業(yè)務(wù)價值

2015-02-11 10:42:48

Apple Watch

2020-12-07 10:10:22

企業(yè)文化客戶體驗(yàn)

2023-04-05 14:19:07

FlinkRedisNoSQL

2016-10-08 00:24:53

httptcp實(shí)時

2021-09-07 10:29:11

JavaScript模塊CSS

2017-05-22 10:33:14

PythonJuliaCython

2013-12-31 09:19:23

Python調(diào)試

2013-12-17 09:02:03

Python調(diào)試

2022-12-21 15:56:23

代碼文檔工具

2023-05-23 13:59:41

RustPython程序

2020-10-11 21:39:35

計算機(jī)互聯(lián)網(wǎng) 技術(shù)

2013-07-26 10:15:29

云計算大數(shù)據(jù)Hadoop

2023-02-15 08:17:20

VSCodeTypeScrip

2021-11-04 17:23:03

Java對象 immutable

2015-10-12 08:56:27

Java不可變

2020-08-25 08:56:55

Pythonawk字符串

2017-11-06 14:18:03

點(diǎn)贊
收藏

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