新聞中心
前言
北宋科學(xué)家沈括在《夢(mèng)溪筆談》第十八卷《技藝》中這樣描述“活字印刷術(shù)”:

慶歷中,有布衣畢昇,又為活版。其法用膠泥刻字,薄如錢(qián)唇,每字為一印,火燒令堅(jiān)……若止印三、二本,未為簡(jiǎn)易;若印數(shù)十百千本,則極為神速。
在日常編碼的過(guò)程中,我們可以總結(jié)出很多“樣板代碼”,就像”活字印刷術(shù)中的“活字”一樣。當(dāng)我們編寫(xiě)新的代碼時(shí),需要用到這些“活字”,就把“樣板代碼”拷貝過(guò)來(lái),修改替換一下就可以了,寫(xiě)起代碼來(lái)“極為神速”?!皹影宕a”其實(shí)就是一種樣例、一種模式、一種經(jīng)驗(yàn)……總結(jié)的“樣板代碼”越多,編寫(xiě)代碼的格式越規(guī)范、質(zhì)量越高、速度越快。
這里,作者總結(jié)了幾種常見(jiàn)Java的“樣板代碼”,希望起到拋磚引玉的作用,希望大家不斷總結(jié)和完善,形成自己的樣板代碼庫(kù)。
一 樣板代碼簡(jiǎn)介
1 什么是樣板代碼?
樣板代碼(Boilerplate Code),通常是指一堆具有固定模式的代碼塊,可以被廣泛地應(yīng)用到各個(gè)程序模塊。
例如,讀取文件就是典型的樣板代碼:
- try (BufferedReader reader = new BufferedReader(new FileReader(fileName))) {
- String line;
- while (Objects.nonNull(line = reader.readLine())) {
- // 處理一行
- ...
- }
- } catch (IOException e) {
- String message = String.format("讀取文件(%s)異常", fileName);
- log.error(message, e);
- throw new ExampleException(message, e);
- }
2 樣板代碼有什么用?
樣板(Boilerplate ),可以拆分為樣例(Example)和模式(Pattern)兩個(gè)單詞進(jìn)行理解——樣例(Example)指可以當(dāng)成一種標(biāo)準(zhǔn)范例,模式(Pattern)指可以作為一種解決方案。當(dāng)遇到類(lèi)似的案例時(shí),就把樣板代碼拷貝過(guò)去,根據(jù)實(shí)際情況進(jìn)行修改,該案例就被輕松解決了。
樣板代碼的主要作用:
- 提供一種標(biāo)準(zhǔn)樣例:可以用于新人學(xué)習(xí),能夠快速上手并使用;
- 提供一種解決方案:遇到類(lèi)似案例時(shí),可以快速利用該方案進(jìn)行解決;
- 有助于不斷積累經(jīng)驗(yàn):當(dāng)發(fā)現(xiàn)一種樣例代碼時(shí),都會(huì)不斷地進(jìn)行優(yōu)化,力求達(dá)到最佳樣例;
- 有助于提高代碼質(zhì)量:樣板代碼必然通過(guò)了時(shí)間考驗(yàn),存在BUG和出錯(cuò)的幾率相對(duì)比較低;
- 有助于提高編碼速度:利用樣板代碼編碼,只是復(fù)制粘貼修改代碼,編碼速度大幅提高;
- 有助于統(tǒng)一代碼樣式:心中有了樣板代碼,就能保證每次都寫(xiě)出統(tǒng)一樣式的代碼。
3 如何編寫(xiě)樣板代碼?
在作者以前的文章《這6種編碼方法,你掌握了幾個(gè)?》中,有詳細(xì)的說(shuō)明和舉例,這里不再累述。其中,適合于樣板代碼的編寫(xiě)方法有:
復(fù)制粘貼生成代碼:利用復(fù)制粘貼樣板代碼,用好了編碼會(huì)事半功倍。
用文本替換生成代碼:利用文本替換生成代碼,可以很快生成一段新代碼。
用Excel公式生成代碼:把樣板代碼先公式化,傳入不同的參數(shù),生成不同的代碼。
用工具或插件生成代碼:很多開(kāi)發(fā)工具或插件都提供一些工具生成代碼,比如:生成構(gòu)造方法、重載基類(lèi)/接口方法、生成Getter/Setter方法、生成toString方法、生成數(shù)據(jù)庫(kù)訪問(wèn)方法……能夠避免很多手敲代碼。
用代碼生成代碼:用代碼生成代碼,就是自己編寫(xiě)代碼,按照自己的樣板代碼格式生成代碼。
4 如何減少樣板代碼?
樣板代碼(Boilerplate Code)具有很大的重復(fù)性,通常被認(rèn)為是一種冗余而又不得不寫(xiě)的代碼。其實(shí)不然,有些樣板代碼不能減少,只是我們還沒(méi)有遇到合適的解決方案而已。通常情況下,我們可以通過(guò)以下幾種方式減少樣板代碼:
利用注解減少樣板代碼
比如,JavaBean模型類(lèi)中的Getter/Setter就是樣板代碼,我們可以通過(guò)Lombok的@Getter/@Setter注解來(lái)減少這樣的樣板代碼。
原始代碼:
- public class User {
- private Long id;
- ...
- public Long getId() {
- return id;
- }
- public void setId(Long id) {
- this.id = id;
- }
- ...
- }
優(yōu)化代碼:
- @Getter
- @Setter
- public class User {
- private Long id;
- ...
- }
利用框架減少樣板代碼
比如,MyBatis 是一款優(yōu)秀的持久層框架,封裝了獲取數(shù)據(jù)庫(kù)連接和聲明、設(shè)置參數(shù)、獲取結(jié)果集等所有JDBC操作。MyBatis 可以通過(guò)簡(jiǎn)單的 XML 或注解來(lái)配置和映射原始類(lèi)型、接口和 Java POJO(Plain Old Java Objects,普通老式 Java 對(duì)象)為數(shù)據(jù)庫(kù)中的記錄。
原始代碼:
- /** 查詢公司員工 */
- public List
queryEmployee(Long companyId) { - try (Connection connection = tddlDataSource.getConnection();
- PreparedStatement statement = connection.prepareStatement(QUERY_EMPLOYEE_SQL)) {
- statement.setLong(1, companyId);
- try (ResultSet result = statement.executeQuery()) {
- List
employeeList = new ArrayList<>(); - while (result.next()) {
- EmployeeDO employee = new EmployeeDO();
- employee.setId(result.getLong(1));
- employee.setName(result.getString(2));
- ...
- employeeList.add(employee);
- }
- return employeeList;
- }
- } catch (SQLException e) {
- String message = String.format("查詢公司(%s)用戶異常", companyId);
- log.error(message, e);
- throw new ExampleException(message, e);
- }
- }
優(yōu)化代碼:
1)UserDAO.java
- @Mapper
- public interface UserDAO {
- List
queryEmployee(@Param("companyId") Long companyId); - }
2)UserDAO.xml
- select id
- , name
- ...
- from t_user
- where company_id = #{companyId}
利用設(shè)計(jì)模式減少樣板代碼
利用設(shè)計(jì)模式,可以把一些重復(fù)性代碼進(jìn)行封裝。比如,上面的讀取文件行模式代碼,就可以用模板方法進(jìn)行封裝。
原始代碼:
- try (BufferedReader reader = new BufferedReader(new FileReader(fileName))) {
- String line;
- while (Objects.nonNull(line = reader.readLine())) {
- // 處理一行
- ...
- }
- } catch (IOException e) {
- String message = String.format("讀取文件(%s)異常", fileName);
- log.error(message, e);
- throw new ExampleException(message, e);
- }
優(yōu)化代碼:
- /** 定義方法 */
- public static void readLine(String fileName, Consumer
lineConsumer) { - try (BufferedReader reader = new BufferedReader(new FileReader(fileName))) {
- String line;
- while (Objects.nonNull(line = reader.readLine())) {
- lineConsumer.accept(line);
- }
- } catch (IOException e) {
- String message = String.format("讀取文件(%s)異常", fileName);
- log.error(message, e);
- throw new ExampleException(message, e);
- }
- }
- // 使用代碼
- readLine("example.txt", line -> {
- // 處理一行
- ...
- });
5 消滅不了的樣板代碼
如果樣板代碼可以被消滅,那么世界上就不存在樣板代碼了。即便是上一節(jié)提供的減少樣板代碼方法,也不能完全的消滅樣板代碼,因?yàn)檫@些樣板代碼依舊存在于框架和模式的實(shí)現(xiàn)中。所以,樣板代碼是消滅不了的。
既然不能消滅樣板代碼,那就應(yīng)該合理地利用樣板代碼。提煉一段樣板代碼,若只用二三次,未為簡(jiǎn)便;若用數(shù)十百千次,則極為神速。下面,列舉了幾種常見(jiàn)Java的樣板代碼,描述了樣板代碼在日常編程中如何提煉和使用。
二 定義工具類(lèi)
1 常用定義方式
通常,我們會(huì)如下定義工具類(lèi):
- /** 例子工具類(lèi) */
- public class ExampleHelper {
- /** 常量值 */
- public final static int CONST_VALUE = 123;
- /** 求和方法 */
- public static int sum(int a, int b) {
- return a + b;
- }
- }
2 存在一些問(wèn)題
修飾符順序不規(guī)范
通過(guò)SonarLint插件掃描,會(huì)出現(xiàn)以下問(wèn)題:
Java語(yǔ)言規(guī)范建議使用”static final”,而不是”final static”。請(qǐng)記住這么一條規(guī)則:靜態(tài)常量,靜態(tài)(static)在前,常量(final)在后。
工具類(lèi)可以被繼承覆蓋
如果我們定義一個(gè)MyExampleHelper來(lái)繼承ExampleHelper:
- public class MyExampleHelper extends ExampleHelper {
- /** 常量值 */
- public static final int CONST_VALUE = 321;
- /** 求和方法 */
- public static int sum(int a, int b) {
- return a * b;
- }
- }
會(huì)發(fā)現(xiàn),MyExampleHelper會(huì)對(duì)ExampleHelper中的常量和方法進(jìn)行覆蓋,導(dǎo)致我們不知道是不是使用了ExampleHelper中的常量和方法。
對(duì)于Apache提供的工具類(lèi),很多同學(xué)都喜歡定義相同名稱(chēng)的工具類(lèi),并讓這個(gè)工具類(lèi)繼承Apache的工具類(lèi),并在這個(gè)類(lèi)中添加自己的實(shí)現(xiàn)方法。其實(shí),我是非常不推薦這種做法的,因?yàn)槟悴恢馈阏{(diào)用的是Apache工具類(lèi)提供的常量和方法,還是被覆蓋的常量和方法。最好的辦法,就是對(duì)工具類(lèi)添加final關(guān)鍵字,讓這個(gè)工具類(lèi)不能被繼承和覆蓋。
工具類(lèi)可以被實(shí)例化
對(duì)于ExampleHelper工具類(lèi),我們可以這樣使用:
- int value = ExampleHelper.CONST_VALUE;
- int sum = ExampleHelper.sum(1, 2);
也可以被這樣使用:
- ExampleHelper exampleHelper = new ExampleHelper();
- int value = exampleHelper.CONST_VALUE;
- int sum = exampleHelper.sum(1, 2);
對(duì)于工具類(lèi)來(lái)說(shuō),沒(méi)有必要進(jìn)行實(shí)例化。所以,我們建議添加私有構(gòu)造方法,并在方法中拋出UnsupportedOperationException(不支持的操作異常)。
3 最佳定義方式
根據(jù)以上存在問(wèn)題及其解決方法,最佳定義的ExampleHelper工具類(lèi)如下:
- /** 例子工具類(lèi) */
- public final class ExampleHelper {
- /** 常量值 */
- public static final int CONST_VALUE = 123;
- /** 構(gòu)造方法 */
- private ExampleHelper() {
- throw new UnsupportedOperationException();
- }
- /** 求和方法 */
- public static int sum(int a, int b) {
- return a + b;
- }
- }
三 定義枚舉類(lèi)
1 常用定義方式
通常,我們會(huì)如下定義枚舉類(lèi):
- /** 例子枚舉類(lèi) */
- public enum ExampleEnum {
- /** 枚舉相關(guān) */
- ONE(1, "one(1)"),
- TWO(2, "two(2)"),
- THREE(3, "two(3)");
- /** 屬性相關(guān) */
- private Integer value;
- private String desc;
- /** 構(gòu)造方法 */
- private ExampleEnum(Integer value, String desc) {
- this.value = value;
- this.desc = desc;
- }
- /** 獲取取值 */
- public Integer getValue() {
- return value;
- }
- /** 獲取描述 */
- public String getDesc() {
- return desc;
- }
- }
2 一些優(yōu)化建議
修飾符private可缺省
通過(guò)SonarLint插件掃描,會(huì)出現(xiàn)以下問(wèn)題:
根據(jù)建議,應(yīng)該刪除構(gòu)造方法前多余的private修飾符。
建議使用基礎(chǔ)類(lèi)型
用包裝類(lèi)型Integer保存枚舉取值,本身并沒(méi)有什么問(wèn)題。但是,本著能用基礎(chǔ)類(lèi)型就用基礎(chǔ)類(lèi)型的規(guī)則,所以建議使用基礎(chǔ)類(lèi)型int。
建議使用final字段
假設(shè),我們要實(shí)現(xiàn)一個(gè)靜態(tài)方法,可能一不小心就把枚舉值給修改了:
- /** 修改取值 */
- public static void modifyValue() {
- for (ExampleEnum value : values()) {
- value.value++;
- }
- }
如果調(diào)用了modifyValue方法,就會(huì)把枚舉值修改,導(dǎo)致應(yīng)用程序出錯(cuò)。為了避免這樣的情況出現(xiàn),我們建議對(duì)字段添加final修飾符,從而避免字段值被惡意篡改。
3 最佳定義方式
- /** 例子枚舉類(lèi) */
- public enum ExampleEnum {
- /** 枚舉相關(guān) */
- ONE(1, "one(1)"),
- TWO(2, "two(2)"),
- THREE(3, "two(3)");
- /** 字段相關(guān) */
- private final int value;
- private final String desc;
- /** 構(gòu)造方法 */
- ExampleEnum(int value, String desc) {
- this.value = value;
- this.desc = desc;
- }
- /** 獲取取值 */
- public int getValue() {
- return value;
- }
- /** 獲取描述 */
- public String getDesc() {
- return desc;
- }
- }
四 定義模型類(lèi)
下面,以定義User(用戶)模型類(lèi)為例,從JavaBean模式、重載構(gòu)造方法、Builder模式3種方式,來(lái)說(shuō)明模型類(lèi)的定義方法以及優(yōu)缺點(diǎn)。
假設(shè):User(用戶)模型類(lèi)共有4個(gè)屬性——id(標(biāo)識(shí))、name(名稱(chēng))、age(年齡)、desc(描述),其中必填屬性為——id(標(biāo)識(shí))、name(名稱(chēng)),可填屬性為——age(年齡)、desc(描述)。
1 JavaBean模式
JavaBean是一個(gè)遵循特定寫(xiě)法的Java類(lèi),它通常具有如下特點(diǎn):
- 必須具有一個(gè)無(wú)參的構(gòu)造方法;
- 所有屬性字段必須是私有的;
- 所有屬性字段必須通過(guò)遵循一種命名規(guī)范的Getter/Setter方法開(kāi)放出來(lái)。
通過(guò)JavaBean模式定義的User(用戶)模型類(lèi)如下:
- /** 用戶類(lèi) */
- public class User {
- private Long id;
- private String name;
- private Integer age;
- private String desc;
- public Long getId() {return id;}
- public void setId(Long id) {this.id = id;}
- public String getName() {return name;}
- public void setName(String name) {this.name = name;}
- public Integer getAge() {return age;}
- public vid setAge(Integer age) {this.age = age;}
- public String getDesc() {return desc;}
- public void setDesc(String desc) {this.desc = desc;}
- }
注意:也可以通過(guò)Lombok的@Getter/@Setter注解生成對(duì)應(yīng)個(gè)Getter/Setter方法。
使用代碼:
- User user = new User();
- user.setId(1L);
- user.setName("alibaba");
- user.setAge(102);
- user.setDesc("test");
- verifyUser(user);
主要優(yōu)點(diǎn):
- 代碼非常簡(jiǎn)單,只有私有屬性字段及其公有Getter/Setter方法;
- 賦值對(duì)象代碼可讀性較強(qiáng),明確地知道哪個(gè)屬性字段對(duì)應(yīng)哪個(gè)值;
- 非常簡(jiǎn)單實(shí)用,被廣泛地用于HSF、Dubbo、MyBatis等中間件。
主要缺點(diǎn):
- 由于可以通過(guò)Setter方法設(shè)置屬性字段,所以不能定義為不可變類(lèi);
- 由于每個(gè)字段分別設(shè)置,所以不能保證字段必填,必須設(shè)置完畢后進(jìn)行統(tǒng)一驗(yàn)證。
2 重載構(gòu)造方法
通過(guò)”重載構(gòu)造方法”定義User(用戶)模型類(lèi)如下:
- /** 用戶類(lèi) */
- public final class User {
- private Long id;
- private String name;
- private Integer age;
- private String desc;
- public User(Long id, String name) {
- this(id, name, null);
- }
- public User(Long id, String name, Integer age) {
- this(id, name, age, null);
- }
- public User(Long id, String name, Integer age, String desc) {
- Assert.notNull(id, "標(biāo)識(shí)不能為空");
- Assert.notNull(name, "名稱(chēng)不能為空");
- this.id = id;
- this.name = name;
- this.age = age;
- this.desc = desc;
- }
- public Long getId() {return id;}
- public String getName() {return name;}
- public Integer getAge() {return age;}
- public String getDesc() {return desc;}
- }
使用代碼:
- User user1 = new User(1L, "alibaba");
- User user2 = new User(1L, "alibaba", 102, "test");
主要優(yōu)點(diǎn):
- 初始化對(duì)象代碼簡(jiǎn)潔,只有簡(jiǎn)單的一行代碼;
- 可以定義為不可變類(lèi),初始化后屬性字段值不可變更;
- 可以在構(gòu)造方法內(nèi)進(jìn)行不可空驗(yàn)證。
主要缺點(diǎn):
- 重載構(gòu)造方法數(shù)量過(guò)多,無(wú)法覆蓋必填字段和非必填字段的所有組合;
- 初始化對(duì)象代碼可讀性差,無(wú)法看出哪個(gè)屬性字段對(duì)應(yīng)哪個(gè)值;
- 如果刪除某個(gè)字段,初始化對(duì)象代碼可能不會(huì)報(bào)錯(cuò),導(dǎo)致出現(xiàn)賦值錯(cuò)誤問(wèn)題。
3 Builder模式
- /** 用戶類(lèi) */
- public final class User {
- private Long id;
- private String name;
- private Integer age;
- private String desc;
- private User(Builder builder) {
- this.id = builder.id;
- this.name = builder.name;
- this.age = builder.age;
- this.desc = builder.desc;
- }
- public static Builder newBuilder(Long id, String name) {
- return new Builder(id, name);
- }
- public Long getId() {return id;}
- public String getName() {return name;}
- public Integer getAge() {return age;}
- public String getDesc() {return desc;}
- public static class Builder {
- private Long id;
- private String name;
- private Integer age;
- private String desc;
- private Builder(Long id, String name) {
- Assert.notNull(id, "標(biāo)識(shí)不能為空");
- Assert.notNull(name, "名稱(chēng)不能為空");
- this.id = id;
- this.name = name;
- }
- public Builder age(Integer age) {
- this.age = age;
- return this;
- }
- public Builder desc(String desc) {
- this.desc = desc;
- return this;
- }
- public User build() {
- return new User(this);
- }
- }
- }
注意:可以采用Lombok的@Builder注解簡(jiǎn)化代碼。
使用代碼:
- User user = User.newBuilder(1L, "alibaba").age(102).desc("test").build();
主要優(yōu)點(diǎn):
- 明確了必填參數(shù)和可選參數(shù),在構(gòu)造方法中進(jìn)行驗(yàn)證;
- 可以定義為不可變類(lèi),初始化后屬性字段值不可變更;
- 賦值代碼可讀性較好,明確知道哪個(gè)屬性字段對(duì)應(yīng)哪個(gè)值;
- 支持鏈?zhǔn)椒椒ㄕ{(diào)用,相比于調(diào)用Setter方法,代碼更簡(jiǎn)潔。
主要缺點(diǎn):
- 代碼量較大,多定義了一個(gè)Builder類(lèi),多定義了一套屬性字段,多實(shí)現(xiàn)了一套賦值方法;
- 運(yùn)行效率低,需要先創(chuàng)建Builder實(shí)例,再賦值屬性字段,再創(chuàng)建目標(biāo)實(shí)例,最后拷貝屬性字段。
五 定義集合常量
在編碼中,經(jīng)常使用到各種集合常量,比如List(列表)常量、Set(集合)常量、Map(映射)常量等。
1 普通定義方式
定義代碼:
最簡(jiǎn)單的方法,就是直接定義一個(gè)普通的集合常量。
- /** 例子工具類(lèi) */
- public final class ExampleHelper {
- /** 常量值列表 */
- public static final List
CONST_VALUE_LIST = Arrays.asList(1, 2, 3); - /** 常量值集合 */
- public static final Set
CONST_VALUE_SET = new HashSet<>(Arrays.asList(1, 2, 3)); - /** 常量值映射 */
- public static final Map
CONST_VALUE_MAP; - static {
- CONST_VALUE_MAP = new HashMap<>(MapHelper.DEFAULT);
- CONST_VALUE_MAP.put(1, "value1");
- CONST_VALUE_MAP.put(2, "value2");
- CONST_VALUE_MAP.put(3, "value3");
- }
- ...
- }
使用代碼:
使用也很方便,直接通過(guò)”類(lèi)名.常量名”使用。
- // 使用常量值集合
- List
constValueList = ExampleHelper.CONST_VALUE_LIST; - Set
constValueSet = ExampleHelper.CONST_VALUE_SET; - Map
constValueMap = ExampleHelper.CONST_VALUE_MAP
2 存在主要問(wèn)題
通過(guò)SonarLint插件掃描,會(huì)出現(xiàn)以下問(wèn)題:
由于普通的集合對(duì)象(如ArrayList、HashMap、HashSet等)都是可變集合對(duì)象,即便是定義為靜態(tài)常量,也可以通過(guò)操作方法進(jìn)行修改。所以,上面方法定義的集合常量,并不是真正意義上的集合常量。其中,Arrays.asList方法生成的內(nèi)部ArrayList不能執(zhí)行add/remove/clear方法,但是可以set方法,也屬于可變集合對(duì)象。
- // 操作常量列表
- ExampleHelper.CONST_VALUE_LIST.remove(3); // UnsupportedOperationException
- ExampleHelper.CONST_VALUE_LIST.add(4); // UnsupportedOperationException
- ExampleHelper.CONST_VALUE_LIST.set(1, 20); // [1,20,3]
- ExampleHelper.CONST_VALUE_LIST.clear(); // UnsupportedOperationException
- // 操作常量集合
- ExampleHelper.CONST_VALUE_SET.remove(3); // [1,2]
- ExampleHelper.CONST_VALUE_SET.add(3); // [1,2,3]
- ExampleHelper.CONST_VALUE_SET.clear(); // []
- // 操作常量映射
- ExampleHelper.CONST_VALUE_MAP.remove(3); // {1:"value1",2:"value2"}
- ExampleHelper.CONST_VALUE_MAP.put(3, "value3"); // {1:"value1",2:"value2",3:"value3"}
- ExampleHelper.CONST_VALUE_MAP.clear(); // []
3 最佳定義方式
在JDK中,Collections工具類(lèi)中提供一套方法,用于把可變集合對(duì)象變?yōu)椴豢勺?不可修改,修改時(shí)會(huì)拋出UnsupportedOperationException異常)集合對(duì)象。所以,可以利用這套方法定義集合靜態(tài)常量。
- /** 例子工具類(lèi) */
- public final class ExampleHelper {
- /** 常量值列表 */
- public static final List
CONST_VALUE_LIST = Collections.unmodifiableList(Arrays.asList(1, 2, 3)); - /** 常量值集合 */
- public static final Set
CONST_VALUE_SET = Collections.unmodifiableSet(new HashSet<>(Arrays.asList(1, 2, 3))); - /** 常量值映射 */
- public static final Map
CONST_VALUE_MAP; - static {
- Map
valueMap = new HashMap<>(MapHelper.DEFAULT); - valueMap.put(1, "value1");
- valueMap.put(2, "value2");
- valueMap.put(3, "value3");
- CONST_VALUE_MAP = Collections.unmodifiableMap(valueMap);
- }
- ...
- }
六 定義數(shù)組常量
上一章介紹了如何定義集合常量,這一章就來(lái)介紹一下如何定義數(shù)組常量。
1 定義公有數(shù)組常量
定義代碼:
一般人定義數(shù)組常量,就會(huì)像下面代碼一樣,定義一個(gè)公有數(shù)組常量。
- /** 例子工具類(lèi) */
- public final class ExampleHelper {
- /** 常量值數(shù)組 */
- public static final int[] CONST_VALUES = new int[] {1, 2, 3};
- ...
- }
使用代碼:
使用也很方便,直接通過(guò)”類(lèi)名.常量名”使用。
- // 使用常量值數(shù)組
- int[] constValues = ExampleHelper.CONST_VALUES;
存在問(wèn)題:
但是,可以通過(guò)下標(biāo)修改數(shù)組值,導(dǎo)致數(shù)組常量的值可變。所以,這種方法定義的數(shù)組常量,并不是一個(gè)真正意義上的數(shù)組常量。
- // 修改常量值數(shù)組
- ExampleHelper.CONST_VALUES[1] = 20; // [1, 20, 3]
2 定義公有集合常量
定義代碼:
可以通過(guò)上一章定義集合常量的方法,返回一個(gè)公有集合常量。
- /** 例子工具類(lèi) */
- public final class ExampleHelper {
- /** 常量值列表 */
- public static final List
CONST_VALUE_LIST = - Collections.unmodifiableList(Arrays.asList(1, 2, 3));
- ...
- }
使用代碼:
要想得到數(shù)組常量,就把集合常量轉(zhuǎn)化為數(shù)組常量。
- // 使用常量值列表
- int[] constValues = ExampleHelper.CONST_VALUE_LIST.stream()
- .mapToInt(Integer::intValue).toArray();
存在問(wèn)題:
每一次都會(huì)把集合常量轉(zhuǎn)化為數(shù)組常量,導(dǎo)致程序運(yùn)行效率降低。
3 最佳定義方式
最佳法”私有數(shù)組常量+公有克隆方法”的解決方案。如下代碼所示:先定義一個(gè)私有數(shù)組常量,保證不會(huì)被外部類(lèi)使用;在定義一個(gè)獲取數(shù)組常量方法,并返回一個(gè)數(shù)組常量的克隆值。
定義代碼:
這里,提供一個(gè)”私有數(shù)組常量+公有克隆方法”的解決方案。如下代碼所示:先定義一個(gè)私有數(shù)組常量,保證不會(huì)被外部類(lèi)使用;在定義一個(gè)獲取數(shù)組常量方法,并返回一個(gè)數(shù)組常量的克隆值。
- /** 例子工具類(lèi) */
- public final class ExampleHelper {
- /** 常量值數(shù)組 */
- private static final int[] CONST_VALUES = new int[] {1, 2, 3};
- /** 獲取常量值數(shù)組方法 */
- public static int[] getConstValues() {
- return CONST_VALUES.clone();
- }
- ...
- }
使用代碼:
由于每次返回的是一個(gè)克隆數(shù)組,即便修改了克隆數(shù)組的常量值,也不會(huì)導(dǎo)致原始數(shù)組常量值的修改。
- // 使用常量值方法
- int[] constValues = ExampleHelper.getConstValues(); // [1, 2, 3]
- constValues[1] = 20; // [1, 20, 3]
- constValues = ExampleHelper.getConstValues(); // [1, 2, 3]
七 定義多條件表達(dá)式
1 利用運(yùn)算符&&(或||)直接拼接
定義代碼:
有時(shí)候,我們會(huì)判斷很多條件,需求用&&(或||)連接多個(gè)條件表達(dá)式。
- /** 獲取審核結(jié)果方法 */
- private static Integer getAuditResult(AuditDataVO data) {
- if (isPassed(data.getAuditItem1())
- && isPassed(data.getAuditItem2())
- ...
- && isPassed(data.getAuditItem11())) {
- return AuditResult.PASSED;
- }
- return AuditResult.REJECTED;
- }
存在問(wèn)題:
通過(guò)SonarLint插件掃描,會(huì)存在2個(gè)問(wèn)題:
其中,圈復(fù)雜度(Cyclomatic complexity,CC)也稱(chēng)為條件復(fù)雜度,是一種衡量代碼復(fù)雜度的標(biāo)準(zhǔn),其符號(hào)為V(G)。
麥凱布最早提出一種稱(chēng)為“基礎(chǔ)路徑測(cè)試”(Basis Path Testing)的軟件測(cè)試方式,測(cè)試程序中的每一線性獨(dú)立路徑,所需的測(cè)試用例個(gè)數(shù)即為程序的圈復(fù)雜度。
圈復(fù)雜度可以用來(lái)衡量一個(gè)模塊判定結(jié)構(gòu)的復(fù)雜程度,其數(shù)量上表現(xiàn)為獨(dú)立路徑的條數(shù),也可理解為覆蓋所有的可能情況最少使用的測(cè)試用例個(gè)數(shù)。
2 利用運(yùn)算符=和&&(或||)級(jí)聯(lián)拼接
定義代碼:
那么,就把&&(或||)連接符拆開(kāi),利用運(yùn)算符=和&&(或||)級(jí)聯(lián)進(jìn)行拼接。
- /** 獲取審核結(jié)果方法 */
- private static AuditResult getAuditResult(AuditDataVO data) {
- boolean isPassed = isPassed(data.getAuditItem1());
- isPassed = isPassed && isPassed(data.getAuditItem2());
- ...
- isPassed = isPassed && isPassed(data.getAuditItem11());
- if (isPassed) {
- return AuditResult.PASSED;
- }
- return AuditResult.REJECTED;
- }
存在問(wèn)題:
通過(guò)SonarLint插件掃描,還存在1個(gè)問(wèn)題:
也就是,利用運(yùn)算符=和&&(或||)級(jí)聯(lián)進(jìn)行拼接,并不能減少方法的圈復(fù)雜度。
3 利用動(dòng)態(tài)無(wú)參數(shù)Lambda表達(dá)式列表
定義代碼:
下面,利用動(dòng)態(tài)無(wú)參數(shù)Lambda表達(dá)式列表優(yōu)化,即把每個(gè)條件表達(dá)式作為BooleanSupplier對(duì)象存在列表中,然后依次執(zhí)行條件表達(dá)式得出最后結(jié)果。
- /** 獲取審核結(jié)果方法 */
- private static AuditResult getAuditResult(AuditDataVO data) {
- List
supplierList = new ArrayList<>(); - supplierList.add(() -> isPassed(data.getAuditItem1()));
- supplierList.add(() -> isPassed(data.
分享名稱(chēng):Java編程技巧之樣板代碼
網(wǎng)站路徑:http://fisionsoft.com.cn/article/dpoidsj.html


咨詢
建站咨詢
