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

Java編程技巧之樣板代碼

開發(fā) 開發(fā)工具
在日常編碼的過程中,我們可以總結出很多“樣板代碼”,就像”活字印刷術中的“活字”一樣。當我們編寫新的代碼時,需要用到這些“活字”,就把“樣板代碼”拷貝過來,修改替換一下就可以了,寫起代碼來“極為神速”。

[[404974]]

前言

北宋科學家沈括在《夢溪筆談》第十八卷《技藝》中這樣描述“活字印刷術”:

慶歷中,有布衣畢昇,又為活版。其法用膠泥刻字,薄如錢唇,每字為一印,火燒令堅……若止印三、二本,未為簡易;若印數(shù)十百千本,則極為神速。

在日常編碼的過程中,我們可以總結出很多“樣板代碼”,就像”活字印刷術中的“活字”一樣。當我們編寫新的代碼時,需要用到這些“活字”,就把“樣板代碼”拷貝過來,修改替換一下就可以了,寫起代碼來“極為神速”。“樣板代碼”其實就是一種樣例、一種模式、一種經(jīng)驗……總結的“樣板代碼”越多,編寫代碼的格式越規(guī)范、質(zhì)量越高、速度越快。

這里,作者總結了幾種常見Java的“樣板代碼”,希望起到拋磚引玉的作用,希望大家不斷總結和完善,形成自己的樣板代碼庫。

一 樣板代碼簡介

1 什么是樣板代碼?

樣板代碼(Boilerplate Code),通常是指一堆具有固定模式的代碼塊,可以被廣泛地應用到各個程序模塊。

例如,讀取文件就是典型的樣板代碼:

  1. try (BufferedReader reader = new BufferedReader(new FileReader(fileName))) { 
  2.     String line; 
  3.     while (Objects.nonNull(line = reader.readLine())) { 
  4.         // 處理一行 
  5.         ... 
  6.     } 
  7. } catch (IOException e) { 
  8.     String message = String.format("讀取文件(%s)異常", fileName); 
  9.     log.error(message, e); 
  10.     throw new ExampleException(message, e); 

2 樣板代碼有什么用?

樣板(Boilerplate ),可以拆分為樣例(Example)和模式(Pattern)兩個單詞進行理解——樣例(Example)指可以當成一種標準范例,模式(Pattern)指可以作為一種解決方案。當遇到類似的案例時,就把樣板代碼拷貝過去,根據(jù)實際情況進行修改,該案例就被輕松解決了。

樣板代碼的主要作用:

  1. 提供一種標準樣例:可以用于新人學習,能夠快速上手并使用;
  2. 提供一種解決方案:遇到類似案例時,可以快速利用該方案進行解決;
  3. 有助于不斷積累經(jīng)驗:當發(fā)現(xiàn)一種樣例代碼時,都會不斷地進行優(yōu)化,力求達到最佳樣例;
  4. 有助于提高代碼質(zhì)量:樣板代碼必然通過了時間考驗,存在BUG和出錯的幾率相對比較低;
  5. 有助于提高編碼速度:利用樣板代碼編碼,只是復制粘貼修改代碼,編碼速度大幅提高;
  6. 有助于統(tǒng)一代碼樣式:心中有了樣板代碼,就能保證每次都寫出統(tǒng)一樣式的代碼。

3 如何編寫樣板代碼?

在作者以前的文章《這6種編碼方法,你掌握了幾個?》中,有詳細的說明和舉例,這里不再累述。其中,適合于樣板代碼的編寫方法有:

復制粘貼生成代碼:利用復制粘貼樣板代碼,用好了編碼會事半功倍。

用文本替換生成代碼:利用文本替換生成代碼,可以很快生成一段新代碼。

用Excel公式生成代碼:把樣板代碼先公式化,傳入不同的參數(shù),生成不同的代碼。

用工具或插件生成代碼:很多開發(fā)工具或插件都提供一些工具生成代碼,比如:生成構造方法、重載基類/接口方法、生成Getter/Setter方法、生成toString方法、生成數(shù)據(jù)庫訪問方法……能夠避免很多手敲代碼。

用代碼生成代碼:用代碼生成代碼,就是自己編寫代碼,按照自己的樣板代碼格式生成代碼。

4 如何減少樣板代碼?

樣板代碼(Boilerplate Code)具有很大的重復性,通常被認為是一種冗余而又不得不寫的代碼。其實不然,有些樣板代碼不能減少,只是我們還沒有遇到合適的解決方案而已。通常情況下,我們可以通過以下幾種方式減少樣板代碼:

利用注解減少樣板代碼

比如,JavaBean模型類中的Getter/Setter就是樣板代碼,我們可以通過Lombok的@Getter/@Setter注解來減少這樣的樣板代碼。

原始代碼:

  1. public class User { 
  2.     private Long id; 
  3.     ... 
  4.     public Long getId() { 
  5.         return id; 
  6.     } 
  7.     public void setId(Long id) { 
  8.         this.id = id; 
  9.     } 
  10.     ... 

優(yōu)化代碼:

  1. @Getter 
  2. @Setter 
  3. public class User { 
  4.     private Long id; 
  5.     ... 

利用框架減少樣板代碼

比如,MyBatis 是一款優(yōu)秀的持久層框架,封裝了獲取數(shù)據(jù)庫連接和聲明、設置參數(shù)、獲取結果集等所有JDBC操作。MyBatis 可以通過簡單的 XML 或注解來配置和映射原始類型、接口和 Java POJO(Plain Old Java Objects,普通老式 Java 對象)為數(shù)據(jù)庫中的記錄。

原始代碼:

  1. /** 查詢公司員工 */ 
  2. public List<EmployeeDO> queryEmployee(Long companyId) { 
  3.     try (Connection connection = tddlDataSource.getConnection(); 
  4.         PreparedStatement statement = connection.prepareStatement(QUERY_EMPLOYEE_SQL)) { 
  5.         statement.setLong(1, companyId); 
  6.         try (ResultSet result = statement.executeQuery()) { 
  7.             List<EmployeeDO> employeeList = new ArrayList<>(); 
  8.             while (result.next()) { 
  9.                 EmployeeDO employee = new EmployeeDO(); 
  10.                 employee.setId(result.getLong(1)); 
  11.                 employee.setName(result.getString(2)); 
  12.                 ... 
  13.                 employeeList.add(employee); 
  14.             } 
  15.             return employeeList; 
  16.         } 
  17.     } catch (SQLException e) { 
  18.         String message = String.format("查詢公司(%s)用戶異常", companyId); 
  19.         log.error(message, e); 
  20.         throw new ExampleException(message, e); 
  21.     } 

優(yōu)化代碼:

1)UserDAO.java

  1. @Mapper 
  2. public interface UserDAO { 
  3.     List<EmployeeDO> queryEmployee(@Param("companyId") Long companyId); 

2)UserDAO.xml

  1. <mapper namespace="com.example.repository.UserDAO"
  2.     <select id="queryEmployee" resultType="com.example.repository.UserDO"
  3.         select id 
  4.         , name 
  5.         ... 
  6.         from t_user 
  7.         where company_id = #{companyId} 
  8.     </select
  9. </mapper> 

 

利用設計模式減少樣板代碼

利用設計模式,可以把一些重復性代碼進行封裝。比如,上面的讀取文件行模式代碼,就可以用模板方法進行封裝。

原始代碼:

  1. try (BufferedReader reader = new BufferedReader(new FileReader(fileName))) { 
  2.     String line; 
  3.     while (Objects.nonNull(line = reader.readLine())) { 
  4.         // 處理一行 
  5.         ... 
  6.     } 
  7. } catch (IOException e) { 
  8.     String message = String.format("讀取文件(%s)異常", fileName); 
  9.     log.error(message, e); 
  10.     throw new ExampleException(message, e); 

優(yōu)化代碼:

  1. /** 定義方法 */ 
  2. public static void readLine(String fileName, Consumer<String> lineConsumer) { 
  3.     try (BufferedReader reader = new BufferedReader(new FileReader(fileName))) { 
  4.         String line; 
  5.         while (Objects.nonNull(line = reader.readLine())) { 
  6.             lineConsumer.accept(line); 
  7.         } 
  8.     } catch (IOException e) { 
  9.         String message = String.format("讀取文件(%s)異常", fileName); 
  10.         log.error(message, e); 
  11.         throw new ExampleException(message, e); 
  12.     } 
  13.  
  14. // 使用代碼 
  15. readLine("example.txt", line -> { 
  16.     // 處理一行 
  17.     ... 
  18. }); 

5 消滅不了的樣板代碼

如果樣板代碼可以被消滅,那么世界上就不存在樣板代碼了。即便是上一節(jié)提供的減少樣板代碼方法,也不能完全的消滅樣板代碼,因為這些樣板代碼依舊存在于框架和模式的實現(xiàn)中。所以,樣板代碼是消滅不了的。

既然不能消滅樣板代碼,那就應該合理地利用樣板代碼。提煉一段樣板代碼,若只用二三次,未為簡便;若用數(shù)十百千次,則極為神速。下面,列舉了幾種常見Java的樣板代碼,描述了樣板代碼在日常編程中如何提煉和使用。

二 定義工具類

1 常用定義方式

通常,我們會如下定義工具類:

  1. /** 例子工具類 */ 
  2. public class ExampleHelper { 
  3.     /** 常量值 */ 
  4.     public final static int CONST_VALUE = 123; 
  5.     /** 求和方法 */ 
  6.     public static int sum(int a, int b) { 
  7.         return a + b; 
  8.     } 

2 存在一些問題

修飾符順序不規(guī)范

通過SonarLint插件掃描,會出現(xiàn)以下問題:

Java語言規(guī)范建議使用”static final”,而不是”final static”。請記住這么一條規(guī)則:靜態(tài)常量,靜態(tài)(static)在前,常量(final)在后。

工具類可以被繼承覆蓋

如果我們定義一個MyExampleHelper來繼承ExampleHelper:

  1. public class MyExampleHelper extends ExampleHelper { 
  2.     /** 常量值 */ 
  3.     public static final int CONST_VALUE = 321; 
  4.  
  5.     /** 求和方法 */ 
  6.     public static int sum(int a, int b) { 
  7.         return a * b; 
  8.     } 

會發(fā)現(xiàn),MyExampleHelper會對ExampleHelper中的常量和方法進行覆蓋,導致我們不知道是不是使用了ExampleHelper中的常量和方法。

對于Apache提供的工具類,很多同學都喜歡定義相同名稱的工具類,并讓這個工具類繼承Apache的工具類,并在這個類中添加自己的實現(xiàn)方法。其實,我是非常不推薦這種做法的,因為你不知道——你調(diào)用的是Apache工具類提供的常量和方法,還是被覆蓋的常量和方法。最好的辦法,就是對工具類添加final關鍵字,讓這個工具類不能被繼承和覆蓋。

工具類可以被實例化

對于ExampleHelper工具類,我們可以這樣使用:

  1. int value = ExampleHelper.CONST_VALUE; 
  2. int sum = ExampleHelper.sum(1, 2); 

也可以被這樣使用:

  1. ExampleHelper exampleHelper = new ExampleHelper(); 
  2. int value = exampleHelper.CONST_VALUE; 
  3. int sum = exampleHelper.sum(1, 2); 

對于工具類來說,沒有必要進行實例化。所以,我們建議添加私有構造方法,并在方法中拋出UnsupportedOperationException(不支持的操作異常)。

3 最佳定義方式

根據(jù)以上存在問題及其解決方法,最佳定義的ExampleHelper工具類如下:

  1. /** 例子工具類 */ 
  2. public final class ExampleHelper { 
  3.     /** 常量值 */ 
  4.     public static final int CONST_VALUE = 123; 
  5.  
  6.     /** 構造方法 */ 
  7.     private ExampleHelper() { 
  8.         throw new UnsupportedOperationException(); 
  9.     } 
  10.  
  11.     /** 求和方法 */ 
  12.     public static int sum(int a, int b) { 
  13.         return a + b; 
  14.     } 

三 定義枚舉類

1 常用定義方式

通常,我們會如下定義枚舉類:

  1. /** 例子枚舉類 */ 
  2. public enum ExampleEnum { 
  3.     /** 枚舉相關 */ 
  4.     ONE(1, "one(1)"), 
  5.     TWO(2, "two(2)"), 
  6.     THREE(3, "two(3)"); 
  7.  
  8.     /** 屬性相關 */ 
  9.     private Integer value; 
  10.     private String desc
  11.  
  12.     /** 構造方法 */ 
  13.     private ExampleEnum(Integer value, String desc) { 
  14.         this.value = value; 
  15.         this.desc = desc
  16.     } 
  17.  
  18.     /** 獲取取值 */ 
  19.     public Integer getValue() { 
  20.         return value; 
  21.     } 
  22.  
  23.     /** 獲取描述 */ 
  24.     public String getDesc() { 
  25.         return desc
  26.     } 

2 一些優(yōu)化建議

修飾符private可缺省

通過SonarLint插件掃描,會出現(xiàn)以下問題:

根據(jù)建議,應該刪除構造方法前多余的private修飾符。

建議使用基礎類型

用包裝類型Integer保存枚舉取值,本身并沒有什么問題。但是,本著能用基礎類型就用基礎類型的規(guī)則,所以建議使用基礎類型int。

建議使用final字段

假設,我們要實現(xiàn)一個靜態(tài)方法,可能一不小心就把枚舉值給修改了:

  1. /** 修改取值 */ 
  2. public static void modifyValue() { 
  3.     for (ExampleEnum value : values()) { 
  4.         value.value++; 
  5.     } 

如果調(diào)用了modifyValue方法,就會把枚舉值修改,導致應用程序出錯。為了避免這樣的情況出現(xiàn),我們建議對字段添加final修飾符,從而避免字段值被惡意篡改。

3 最佳定義方式

  1. /** 例子枚舉類 */ 
  2. public enum ExampleEnum { 
  3.     /** 枚舉相關 */ 
  4.     ONE(1, "one(1)"), 
  5.     TWO(2, "two(2)"), 
  6.     THREE(3, "two(3)"); 
  7.  
  8.     /** 字段相關 */ 
  9.     private final int value; 
  10.     private final String desc
  11.  
  12.     /** 構造方法 */ 
  13.     ExampleEnum(int value, String desc) { 
  14.         this.value = value; 
  15.         this.desc = desc
  16.     } 
  17.  
  18.     /** 獲取取值 */ 
  19.     public int getValue() { 
  20.         return value; 
  21.     } 
  22.  
  23.     /** 獲取描述 */ 
  24.     public String getDesc() { 
  25.         return desc
  26.     } 

四 定義模型類

下面,以定義User(用戶)模型類為例,從JavaBean模式、重載構造方法、Builder模式3種方式,來說明模型類的定義方法以及優(yōu)缺點。

假設:User(用戶)模型類共有4個屬性——id(標識)、name(名稱)、age(年齡)、desc(描述),其中必填屬性為——id(標識)、name(名稱),可填屬性為——age(年齡)、desc(描述)。

1 JavaBean模式

JavaBean是一個遵循特定寫法的Java類,它通常具有如下特點:

  1. 必須具有一個無參的構造方法;
  2. 所有屬性字段必須是私有的;
  3. 所有屬性字段必須通過遵循一種命名規(guī)范的Getter/Setter方法開放出來。

通過JavaBean模式定義的User(用戶)模型類如下:

  1. /** 用戶類 */ 
  2. public class User { 
  3.     private Long id; 
  4.     private String name
  5.     private Integer age; 
  6.     private String desc
  7.  
  8.     public Long getId() {return id;} 
  9.     public void setId(Long id) {this.id = id;} 
  10.     public String getName() {return name;} 
  11.     public void setName(String name) {this.name = name;} 
  12.     public Integer getAge() {return age;} 
  13.     public vid setAge(Integer age) {this.age = age;} 
  14.     public String getDesc() {return desc;} 
  15.     public void setDesc(String desc) {this.desc = desc;} 

注意:也可以通過Lombok的@Getter/@Setter注解生成對應個Getter/Setter方法。

使用代碼:

  1. User user = new User(); 
  2. user.setId(1L); 
  3. user.setName("alibaba"); 
  4. user.setAge(102); 
  5. user.setDesc("test"); 
  6. verifyUser(user); 

主要優(yōu)點:

  1. 代碼非常簡單,只有私有屬性字段及其公有Getter/Setter方法;
  2. 賦值對象代碼可讀性較強,明確地知道哪個屬性字段對應哪個值;
  3. 非常簡單實用,被廣泛地用于HSF、Dubbo、MyBatis等中間件。

主要缺點:

  • 由于可以通過Setter方法設置屬性字段,所以不能定義為不可變類;
  • 由于每個字段分別設置,所以不能保證字段必填,必須設置完畢后進行統(tǒng)一驗證。

2 重載構造方法

通過”重載構造方法”定義User(用戶)模型類如下:

  1. /** 用戶類 */ 
  2. public final class User { 
  3.     private Long id; 
  4.     private String name
  5.     private Integer age; 
  6.     private String desc
  7.  
  8.     public User(Long id, String name) { 
  9.         this(id, namenull); 
  10.     } 
  11.     public User(Long id, String nameInteger age) { 
  12.         this(id, name, age, null); 
  13.     } 
  14.     public User(Long id, String nameInteger age, String desc) { 
  15.         Assert.notNull(id, "標識不能為空"); 
  16.         Assert.notNull(name"名稱不能為空"); 
  17.         this.id = id; 
  18.         this.name = name
  19.         this.age = age; 
  20.         this.desc = desc
  21.     } 
  22.  
  23.     public Long getId() {return id;} 
  24.     public String getName() {return name;} 
  25.     public Integer getAge() {return age;} 
  26.     public String getDesc() {return desc;} 

使用代碼:

  1. User user1 = new User(1L, "alibaba"); 
  2.  
  3. User user2 = new User(1L, "alibaba", 102, "test"); 

主要優(yōu)點:

  1. 初始化對象代碼簡潔,只有簡單的一行代碼;
  2. 可以定義為不可變類,初始化后屬性字段值不可變更;
  3. 可以在構造方法內(nèi)進行不可空驗證。

主要缺點:

  1. 重載構造方法數(shù)量過多,無法覆蓋必填字段和非必填字段的所有組合;
  2. 初始化對象代碼可讀性差,無法看出哪個屬性字段對應哪個值;
  3. 如果刪除某個字段,初始化對象代碼可能不會報錯,導致出現(xiàn)賦值錯誤問題。

3 Builder模式

  1. /** 用戶類 */ 
  2. public final class User { 
  3.     private Long id; 
  4.     private String name
  5.     private Integer age; 
  6.     private String desc
  7.  
  8.     private User(Builder builder) { 
  9.         this.id = builder.id; 
  10.         this.name = builder.name
  11.         this.age = builder.age; 
  12.         this.desc = builder.desc
  13.     } 
  14.     public static Builder newBuilder(Long id, String name) { 
  15.         return new Builder(id, name); 
  16.     } 
  17.  
  18.     public Long getId() {return id;} 
  19.     public String getName() {return name;} 
  20.     public Integer getAge() {return age;} 
  21.     public String getDesc() {return desc;} 
  22.  
  23.     public static class Builder { 
  24.         private Long id; 
  25.         private String name
  26.         private Integer age; 
  27.         private String desc
  28.  
  29.         private Builder(Long id, String name) { 
  30.             Assert.notNull(id, "標識不能為空"); 
  31.             Assert.notNull(name"名稱不能為空"); 
  32.             this.id = id; 
  33.             this.name = name
  34.         } 
  35.         public Builder age(Integer age) { 
  36.             this.age = age; 
  37.             return this; 
  38.         } 
  39.         public Builder desc(String desc) { 
  40.             this.desc = desc
  41.             return this; 
  42.         } 
  43.         public User build() { 
  44.             return new User(this); 
  45.         } 
  46.     } 

注意:可以采用Lombok的@Builder注解簡化代碼。

使用代碼:

  1. User user = User.newBuilder(1L, "alibaba").age(102).desc("test").build(); 

主要優(yōu)點:

  1. 明確了必填參數(shù)和可選參數(shù),在構造方法中進行驗證;
  2. 可以定義為不可變類,初始化后屬性字段值不可變更;
  3. 賦值代碼可讀性較好,明確知道哪個屬性字段對應哪個值;
  4. 支持鏈式方法調(diào)用,相比于調(diào)用Setter方法,代碼更簡潔。

主要缺點:

  1. 代碼量較大,多定義了一個Builder類,多定義了一套屬性字段,多實現(xiàn)了一套賦值方法;
  2. 運行效率低,需要先創(chuàng)建Builder實例,再賦值屬性字段,再創(chuàng)建目標實例,最后拷貝屬性字段。

五 定義集合常量

在編碼中,經(jīng)常使用到各種集合常量,比如List(列表)常量、Set(集合)常量、Map(映射)常量等。

1 普通定義方式

定義代碼:

最簡單的方法,就是直接定義一個普通的集合常量。

  1. /** 例子工具類 */ 
  2. public final class ExampleHelper { 
  3.     /** 常量值列表 */ 
  4.     public static final List<Integer> CONST_VALUE_LIST = Arrays.asList(1, 2, 3); 
  5.     /** 常量值集合 */ 
  6.     public static final Set<Integer> CONST_VALUE_SET = new HashSet<>(Arrays.asList(1, 2, 3)); 
  7.     /** 常量值映射 */ 
  8.     public static final Map<Integer, String> CONST_VALUE_MAP; 
  9.     static { 
  10.         CONST_VALUE_MAP = new HashMap<>(MapHelper.DEFAULT); 
  11.         CONST_VALUE_MAP.put(1, "value1"); 
  12.         CONST_VALUE_MAP.put(2, "value2"); 
  13.         CONST_VALUE_MAP.put(3, "value3"); 
  14.     } 
  15.     ... 

使用代碼:

使用也很方便,直接通過”類名.常量名”使用。

  1. // 使用常量值集合 
  2. List<Integer> constValueList = ExampleHelper.CONST_VALUE_LIST; 
  3. Set<Integer> constValueSet = ExampleHelper.CONST_VALUE_SET; 
  4. Map<Integer, String> constValueMap = ExampleHelper.CONST_VALUE_MAP 

2 存在主要問題

通過SonarLint插件掃描,會出現(xiàn)以下問題:

由于普通的集合對象(如ArrayList、HashMap、HashSet等)都是可變集合對象,即便是定義為靜態(tài)常量,也可以通過操作方法進行修改。所以,上面方法定義的集合常量,并不是真正意義上的集合常量。其中,Arrays.asList方法生成的內(nèi)部ArrayList不能執(zhí)行add/remove/clear方法,但是可以set方法,也屬于可變集合對象。

  1. // 操作常量列表 
  2. ExampleHelper.CONST_VALUE_LIST.remove(3); // UnsupportedOperationException 
  3. ExampleHelper.CONST_VALUE_LIST.add(4); // UnsupportedOperationException 
  4. ExampleHelper.CONST_VALUE_LIST.set(1, 20); // [1,20,3] 
  5. ExampleHelper.CONST_VALUE_LIST.clear(); // UnsupportedOperationException 
  6.  
  7. // 操作常量集合 
  8. ExampleHelper.CONST_VALUE_SET.remove(3); // [1,2] 
  9. ExampleHelper.CONST_VALUE_SET.add(3); // [1,2,3] 
  10. ExampleHelper.CONST_VALUE_SET.clear(); // [] 
  11.  
  12. // 操作常量映射 
  13. ExampleHelper.CONST_VALUE_MAP.remove(3); // {1:"value1",2:"value2"
  14. ExampleHelper.CONST_VALUE_MAP.put(3, "value3"); // {1:"value1",2:"value2",3:"value3"
  15. ExampleHelper.CONST_VALUE_MAP.clear(); // [] 

3 最佳定義方式

在JDK中,Collections工具類中提供一套方法,用于把可變集合對象變?yōu)椴豢勺?不可修改,修改時會拋出UnsupportedOperationException異常)集合對象。所以,可以利用這套方法定義集合靜態(tài)常量。

  1. /** 例子工具類 */ 
  2. public final class ExampleHelper { 
  3.     /** 常量值列表 */ 
  4.     public static final List<Integer> CONST_VALUE_LIST = Collections.unmodifiableList(Arrays.asList(1, 2, 3)); 
  5.     /** 常量值集合 */ 
  6.     public static final Set<Integer> CONST_VALUE_SET = Collections.unmodifiableSet(new HashSet<>(Arrays.asList(1, 2, 3))); 
  7.     /** 常量值映射 */ 
  8.     public static final Map<Integer, String> CONST_VALUE_MAP; 
  9.     static { 
  10.         Map<Integer, String> valueMap = new HashMap<>(MapHelper.DEFAULT); 
  11.         valueMap.put(1, "value1"); 
  12.         valueMap.put(2, "value2"); 
  13.         valueMap.put(3, "value3"); 
  14.         CONST_VALUE_MAP = Collections.unmodifiableMap(valueMap); 
  15.     } 
  16.     ... 

六 定義數(shù)組常量

上一章介紹了如何定義集合常量,這一章就來介紹一下如何定義數(shù)組常量。

1 定義公有數(shù)組常量

定義代碼:

一般人定義數(shù)組常量,就會像下面代碼一樣,定義一個公有數(shù)組常量。

  1. /** 例子工具類 */ 
  2. public final class ExampleHelper { 
  3.     /** 常量值數(shù)組 */ 
  4.     public static final int[] CONST_VALUES = new int[] {1, 2, 3}; 
  5.     ... 

使用代碼:

使用也很方便,直接通過”類名.常量名”使用。

  1. // 使用常量值數(shù)組 
  2. int[] constValues = ExampleHelper.CONST_VALUES; 

存在問題:

但是,可以通過下標修改數(shù)組值,導致數(shù)組常量的值可變。所以,這種方法定義的數(shù)組常量,并不是一個真正意義上的數(shù)組常量。

  1. // 修改常量值數(shù)組 
  2. ExampleHelper.CONST_VALUES[1] = 20; // [1, 20, 3] 

2 定義公有集合常量

定義代碼:

可以通過上一章定義集合常量的方法,返回一個公有集合常量。

  1. /** 例子工具類 */ 
  2. public final class ExampleHelper { 
  3.     /** 常量值列表 */ 
  4.     public static final List<Integer> CONST_VALUE_LIST = 
  5.         Collections.unmodifiableList(Arrays.asList(1, 2, 3)); 
  6.     ... 

使用代碼:

要想得到數(shù)組常量,就把集合常量轉(zhuǎn)化為數(shù)組常量。

  1. // 使用常量值列表 
  2. int[] constValues = ExampleHelper.CONST_VALUE_LIST.stream() 
  3.     .mapToInt(Integer::intValue).toArray(); 

存在問題:

每一次都會把集合常量轉(zhuǎn)化為數(shù)組常量,導致程序運行效率降低。

3 最佳定義方式

最佳法”私有數(shù)組常量+公有克隆方法”的解決方案。如下代碼所示:先定義一個私有數(shù)組常量,保證不會被外部類使用;在定義一個獲取數(shù)組常量方法,并返回一個數(shù)組常量的克隆值。

定義代碼:

這里,提供一個”私有數(shù)組常量+公有克隆方法”的解決方案。如下代碼所示:先定義一個私有數(shù)組常量,保證不會被外部類使用;在定義一個獲取數(shù)組常量方法,并返回一個數(shù)組常量的克隆值。

  1. /** 例子工具類 */ 
  2. public final class ExampleHelper { 
  3.     /** 常量值數(shù)組 */ 
  4.     private static final int[] CONST_VALUES = new int[] {1, 2, 3}; 
  5.     /** 獲取常量值數(shù)組方法 */ 
  6.     public static int[] getConstValues() { 
  7.         return CONST_VALUES.clone(); 
  8.     } 
  9.     ... 

使用代碼:

由于每次返回的是一個克隆數(shù)組,即便修改了克隆數(shù)組的常量值,也不會導致原始數(shù)組常量值的修改。

  1. // 使用常量值方法 
  2. int[] constValues = ExampleHelper.getConstValues(); // [1, 2, 3] 
  3. constValues[1] = 20; // [1, 20, 3] 
  4. constValues = ExampleHelper.getConstValues(); // [1, 2, 3] 

七 定義多條件表達式

1 利用運算符&&(或||)直接拼接

定義代碼:

有時候,我們會判斷很多條件,需求用&&(或||)連接多個條件表達式。

  1. /** 獲取審核結果方法 */ 
  2. private static Integer getAuditResult(AuditDataVO data) { 
  3.     if (isPassed(data.getAuditItem1()) 
  4.         && isPassed(data.getAuditItem2()) 
  5.         ... 
  6.         && isPassed(data.getAuditItem11())) { 
  7.         return AuditResult.PASSED; 
  8.     } 
  9.     return AuditResult.REJECTED; 

存在問題:

通過SonarLint插件掃描,會存在2個問題:

其中,圈復雜度(Cyclomatic complexity,CC)也稱為條件復雜度,是一種衡量代碼復雜度的標準,其符號為V(G)。

麥凱布最早提出一種稱為“基礎路徑測試”(Basis Path Testing)的軟件測試方式,測試程序中的每一線性獨立路徑,所需的測試用例個數(shù)即為程序的圈復雜度。

圈復雜度可以用來衡量一個模塊判定結構的復雜程度,其數(shù)量上表現(xiàn)為獨立路徑的條數(shù),也可理解為覆蓋所有的可能情況最少使用的測試用例個數(shù)。

2 利用運算符=和&&(或||)級聯(lián)拼接

定義代碼:

那么,就把&&(或||)連接符拆開,利用運算符=和&&(或||)級聯(lián)進行拼接。

  1. /** 獲取審核結果方法 */ 
  2. private static AuditResult getAuditResult(AuditDataVO data) { 
  3.     boolean isPassed = isPassed(data.getAuditItem1()); 
  4.     isPassed = isPassed && isPassed(data.getAuditItem2()); 
  5.     ... 
  6.     isPassed = isPassed && isPassed(data.getAuditItem11()); 
  7.     if (isPassed) { 
  8.         return AuditResult.PASSED; 
  9.     } 
  10.     return AuditResult.REJECTED; 

存在問題:

通過SonarLint插件掃描,還存在1個問題:

也就是,利用運算符=和&&(或||)級聯(lián)進行拼接,并不能減少方法的圈復雜度。

3 利用動態(tài)無參數(shù)Lambda表達式列表

定義代碼:

下面,利用動態(tài)無參數(shù)Lambda表達式列表優(yōu)化,即把每個條件表達式作為BooleanSupplier對象存在列表中,然后依次執(zhí)行條件表達式得出最后結果。

  1. /** 獲取審核結果方法 */ 
  2. private static AuditResult getAuditResult(AuditDataVO data) { 
  3.     List<BooleanSupplier> supplierList = new ArrayList<>(); 
  4.     supplierList.add(() -> isPassed(data.getAuditItem1())); 
  5.     supplierList.add(() -> isPassed(data.getAuditItem2())); 
  6.     ... 
  7.     supplierList.add(() -> isPassed(data.getAuditItem11())); 
  8.     for (BooleanSupplier supplier : supplierList) { 
  9.         if (!supplier.getAsBoolean()) { 
  10.             return AuditResult.REJECTED; 
  11.         } 
  12.     } 
  13.     return AuditResult.PASSED; 

存在問題:

通過SonarLint插件掃描,沒有提示任何問題。但是,每次都動態(tài)添加Lambda表達式,就會導致程序效率低下。那么,有沒有把Lambda表達式靜態(tài)化的方法呢?

4 利用靜態(tài)有參數(shù)Lambda表達式列表

定義代碼:

要想固化Lambda表達式,就必須動態(tài)傳入AuditDataVO對象。這里,采用Predicate來接收Lambda表達式,在Lambda表達式中指定AuditDataVO對象data。然后,在for循環(huán)中,依次指定AuditDataVO對象data,并計算表達式的值。

  1. /** 審核結果斷言列表 */ 
  2. private static final List<Predicate<AuditDataVO>> AUDIT_RESULT_PREDICATE_LIST = 
  3.     Collections.unmodifiableList(Arrays.asList( 
  4.         data -> isPassed(data.getAuditItem1()), 
  5.         data -> isPassed(data.getAuditItem2()), 
  6.         ... 
  7.         data -> isPassed(data.getAuditItem11()))); 
  8.  
  9. /** 獲取審核結果方法 */ 
  10. private static AuditResult getAuditResult(AuditDataVO data) { 
  11.     for (Predicate<AuditDataVO> predicate : AUDIT_RESULT_PREDICATE_LIST) { 
  12.         if (!predicate.test(data)) { 
  13.             return AuditResult.REJECTED; 
  14.         } 
  15.     } 
  16.     return AuditResult.PASSED; 

適用條件:

  • 適合于&&(或||)連接大量條件表達式的情況;
  • 適合于每個條件表達式都需要傳入相同參數(shù)的情況,如果每個條件表達式傳入?yún)?shù)不同,只能使用動態(tài)無參數(shù)Lambda表達式列表方法;
  • 如果需要傳入兩個參數(shù),可以使用BiPredicate類型來接收Lambda表達式;如果需要傳入多個參數(shù),則需要自定義方法接口。

后記

明代思想家王陽明在《傳習錄》中說道:

初種根時,只管栽培灌溉,勿作枝想,勿作葉想,勿作花想,勿作實想。懸想何益?但不忘栽培之功,怕沒有枝葉花實?

 

責任編輯:武曉燕 來源: 51CTO專欄
相關推薦

2018-11-20 10:50:00

Java性能優(yōu)化編程技巧

2009-12-07 16:33:55

WCF 緩存

2017-12-23 14:38:41

Android編程開發(fā)優(yōu)化

2025-04-18 04:22:00

2020-09-23 09:20:58

代碼Java字符串

2022-09-26 00:00:01

java代碼開發(fā)

2010-03-22 10:42:37

Java Socket

2021-01-28 08:12:15

Linux命令技巧

2011-07-10 15:26:54

C++

2011-06-14 11:14:10

性能優(yōu)化代碼

2010-01-11 10:28:51

C++編程

2022-06-13 07:03:25

Go 語言怎么優(yōu)化重

2020-08-23 21:07:16

編程PythonJava

2017-04-06 16:20:27

Python趣味代碼編程

2012-12-25 09:45:08

PythonWeb

2011-05-30 15:29:32

C++

2022-09-19 15:02:24

C語言

2011-07-05 14:42:46

java

2021-03-11 12:33:50

JavaPowerMock技巧

2010-08-13 13:40:47

DB2編程序
點贊
收藏

51CTO技術棧公眾號