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

一段代碼被老大要求重構了六次,我心態(tài)崩了

開發(fā) 后端
在本文中,我們重點介紹了函數(shù)式接口的用途和可用性,我們將研究如何將代碼從開始的樣板代碼現(xiàn)演變?yōu)榛诤瘮?shù)式接口的靈活實現(xiàn)。希望對大家理解函數(shù)式接口有所幫助,謝謝大家。

[[396939]]

本文轉載自微信公眾號「今日Java」,作者麥洛。轉載本文請聯(lián)系今日Java公眾號。  

  • 前言
  • 第一次 按類型篩選瓜類
  • 第二次 按重量篩選瓜類
  • 第三次 按類型和重量篩選瓜類
  • 第四次 將行為作為參數(shù)傳遞
  • 第五次 一次性加了100個過濾條件
  • 第六次 引入泛型
  • 簡而言之Lambda
  • 總結

前言

Hi,大家好,我是麥洛。我又回來啦??

進來給大家八卦一段,看看我自己都去干啥了?話說最近公司接了一個農(nóng)產(chǎn)品交易網(wǎng)站新項目,因為一段代碼重構問題差點和老大干起來,本來以為是老大故意刁難我。最后還是發(fā)現(xiàn)是我太菜了?事情是這個樣子滴!

在周例會上,老大告知我們最近接了一個農(nóng)產(chǎn)品交易平臺,主要用于全省農(nóng)產(chǎn)品線上交易。首當其中,就是要把我們甘肅省的黃河蜜推銷出去,我就被安排賣瓜!嗷,不,賣瓜這個功能我負責開發(fā)?;很快我設計下面的類來定義瓜 Melon 類:

  1. /** 
  2.  * 瓜 
  3.  * @author Milo Lee 
  4.  * @date 2021-04-07 13:21 
  5.  */ 
  6. public class Melon { 
  7.     /**品種*/ 
  8.     private final String type; 
  9.     /**重量*/ 
  10.     private final int weight; 
  11.     /**產(chǎn)地*/ 
  12.     private final String origin; 
  13.  
  14.     public Melon(String type, int weight, String origin) { 
  15.         this.type = type; 
  16.         this.weight = weight; 
  17.         this.origin = origin; 
  18.     } 
  19.     // getters, toString()方法省略 

經(jīng)過一頓CRUD騷操作,寫完了瓜類增刪改查工作,交工下班。

第一次 按類型篩選瓜類

第二天,老大給我提了一個問題,說增加能夠按瓜類型對瓜進行過濾。這不很簡單嗎?于是,于是我創(chuàng)建了一個 Filters 類, 實現(xiàn)了一個filterMelonByType方法

  1. /** 
  2.  * @author Milo Lee 
  3.  * @date 2021-04-07 13:25 
  4.  */ 
  5. public class Filters { 
  6.  
  7.     /** 
  8.      * 根據(jù)類型篩選瓜類 
  9.      * @param melons 瓜類 
  10.      * @param type 類型 
  11.      * @return 
  12.      */ 
  13.     public static List<Melon> filterMelonByType(List<Melon> melons, String type) { 
  14.  
  15.         List<Melon> result = new ArrayList<>(); 
  16.         for (Melon melon: melons) { 
  17.             if (melon != null && type.equalsIgnoreCase(melon.getType())) { 
  18.                 result.add(melon); 
  19.             } 
  20.         } 
  21.         return result; 
  22.     } 

搞定,我們來測試一下:

  1. public static void main(String[] args) { 
  2.     ArrayList<Melon> melons = new ArrayList<>(); 
  3.     melons.add(new Melon("羊角蜜", 1, "泰國")); 
  4.     melons.add(new Melon("西瓜", 2, "三亞")); 
  5.     melons.add(new Melon("黃河蜜", 3, "蘭州")); 
  6.     List<Melon> melonType = Filters.filterMelonByType(melons, "黃河蜜"); 
  7.     melonType.forEach(melon->{ 
  8.     System.out.println("瓜類型:"+melon.getType()); 
  9.     }); 

沒毛病,拿給老大看看去,老大看了我的代碼說:如果我讓你在增加一個按重量篩選瓜類,你打算怎么寫?回去考慮一下吧,這家伙不會故意找我茬吧???

第二次 按重量篩選瓜類

回到座位的我心想,上次我已經(jīng)實現(xiàn)了按類型篩選瓜類,那我給他copy一份改改吧!

如下所示:

  1. /** 
  2.  * 按照重量過濾瓜類 
  3.  * @param melons 
  4.  * @param weight 
  5.  * @return 
  6.  */ 
  7. public static List<Melon> filterMelonByWeight(List<Melon> melons, int weight) { 
  8.  
  9.     List<Melon> result = new ArrayList<>(); 
  10.     for (Melon melon: melons) { 
  11.         if (melon != null && melon.getWeight() == weight) { 
  12.             result.add(melon); 
  13.         } 
  14.     } 
  15.     return result; 
  1. public static void main(String[] args) { 
  2.       ArrayList<Melon> melons = new ArrayList<>(); 
  3.         melons.add(new Melon("羊角蜜", 1, "泰國")); 
  4.         melons.add(new Melon("西瓜", 2, "三亞")); 
  5.         melons.add(new Melon("黃河蜜", 3, "蘭州")); 
  6.         List<Melon> melonType = Filters.filterMelonByType(melons, "黃河蜜"); 
  7.         melonType.forEach(melon->{ 
  8.             System.out.println("瓜類型:"+melon.getType()); 
  9.         }); 
  10.  
  11.         List<Melon> melonWeight = Filters.filterMelonByWeight( melons,3); 
  12.         melonWeight.forEach(melon->{ 
  13.             System.out.println("瓜重量:"+melon.getWeight()); 
  14.         }); 
  15.     } 

程序員最喜歡的方式,CV搞定,哈哈。但是我發(fā)現(xiàn)filterByWeight()與 filterByType() 非常相似,就是過濾條件不同。我心想,老大不會讓我寫一個按類型和重量篩選瓜類吧。拿著我的代碼去給老大看,果不其然,怕什么來什么。

第三次 按類型和重量篩選瓜類

為了滿足完成老大的任務,我將上面的代碼進行了糅合,很快寫了如下的代碼:

  1. /** 
  2.  * 按照類型和重量來篩選瓜類 
  3.  * @param melons 
  4.  * @param type 
  5.  * @param weight 
  6.  * @return 
  7.  */ 
  8. public static List<Melon> filterMelonByTypeAndWeight(List<Melon> melons, String type, int weight) { 
  9.  
  10.     List<Melon> result = new ArrayList<>(); 
  11.     for (Melon melon: melons) { 
  12.         if (melon != null && type.equalsIgnoreCase(melon.getType()) && melon.getWeight() == weight) { 
  13.             result.add(melon); 
  14.         } 
  15.     } 
  16.     return result; 

老大看了我的代碼說,看來你還是沒有理解我的意思。假如今天不光我,還有客戶繼續(xù)這樣提需求。

那么 Filters 將會有很多這樣類似的方法,也就是說寫了很多樣板代碼(代碼冗余但又不得不寫);

在我們程序員看來,這是不能接受的。如果繼續(xù)添加新的過濾條件,則代碼將變得難以維護且容易出錯。你去了解一下lambda表達式和函數(shù)式接口知識點,再修改一下你的代碼。我已經(jīng)確定了,他就是和我過不去?

第四次 將行為作為參數(shù)傳遞

經(jīng)過上面的三番折騰。我發(fā)現(xiàn)理論上Melon類的任何屬性都有可能作為過濾條件,這樣的話我們的Filter類將會有大量的樣板代碼,而且有些方法會非常復雜。

其實我們可以發(fā)現(xiàn),我們每寫一個方法,都對應一種查詢行為,查詢行為必然對應一種過濾條件。有沒有辦法我們寫一個方法,將查詢行為作為參數(shù)傳遞進去,從而返回我們的結果呢?

那么給它取了一個名字:行為參數(shù)化,在下圖中進行了說明(左側顯示了我們現(xiàn)在擁有的;右側顯示了我們想要的),有沒有發(fā)現(xiàn)樣板代碼會明顯減少??

如果我們將過濾條件視為一種行為,那么將每種行為視為接口的實現(xiàn)是非常直觀的。經(jīng)過分析我們發(fā)現(xiàn)以上所有這些行為都有一個共同點:過濾條件和boolean 類型的返回 。抽象一個接口如下:

  1. public interface MelonPredicate { 
  2.   boolean test(Melon melon); 

例如,過濾 黃河蜜可以這樣寫: HHMMelonPredicate。

  1. public class HHMMelonPredicate implements MelonPredicate { 
  2.  
  3.      @Override 
  4.      public boolean test(Melon melon) { 
  5.        return "黃河蜜".equalsIgnoreCase(melon.getType()); 
  6.  
  7.      } 
  8.  

以此類推,我們也可以過濾一定重量的瓜:

  1. public class WeightMelonPredicate implements MelonPredicate { 
  2.  
  3.      @Override 
  4.      public boolean test(Melon melon) { 
  5.        return melon.getWeight() > 5000; 
  6.      } 
  7.  

其實熟悉設計模式的同學應該知道這就是:策略設計模式。

主要思想就是讓系統(tǒng)在運行時動態(tài)選擇需要調(diào)用的方法。所以我們可以認為 MelonPredicate 接口統(tǒng)一了所有專用于篩選瓜類的算法,并且每種實現(xiàn)都是一種策略,我們也可以把它理解為一種行為。

目前,我們利用策略設計模式,將查詢行為進行了抽象。我們還需要一個方法接收 MelonPredicate 參數(shù)。于是我定義了 filterMelons() 方法,如下所示:

  1. public static List<Melon> filterMelons(List<Melon> melons, MelonPredicate predicate) { 
  2.      
  3.     List<Melon> result = new ArrayList<>(); 
  4.     for (Melon melon: melons) { 
  5.       if (melon != null && predicate.test(melon)) { 
  6.         result.add(melon); 
  7.       } 
  8.  
  9.     }   
  10.     return result; 

大功告成,測試一下,果然比之前好用多了,再讓老大瞅瞅去

  1. List<Melon> hhm = Filters.filterMelons(melons, new HHMMelonPredicate()); 
  2.  
  3. List<Melon> weight = Filters.filterMelons(melons, new WeightMelonPredicate()); 

第五次 一次性加了100個過濾條件

就在我沾沾自喜時候,老大又給他潑了一盆冷水。他說你以為我們的平臺就買黃河蜜啊,如果前前后后有幾十種瓜品種,我給你列出100種過濾條件,你怎么辦?

我的心里一萬個草泥馬在奔騰啊??!老大是不是存心和我過不去啊!雖然經(jīng)過上次改造,我的代碼已經(jīng)足夠靈活,但是如果突然增加100個過濾條件,我仍然需要編寫100個策略類來實現(xiàn)每一個過濾條件。然后我們需要將策略傳遞給 filterMelons() 方法。

有沒有不需要創(chuàng)建這些類的辦法那?聰明的我很快發(fā)現(xiàn)可以使用java匿名內(nèi)部類。

如下所示:

  1. List<Melon> europeans = Filters.filterMelons(melons, new MelonPredicate() { 
  2.  
  3.      @Override 
  4.  
  5.      public boolean test(Melon melon) { 
  6.  
  7.        return "europe".equalsIgnoreCase(melon.getOrigin()); 
  8.  
  9.      } 
  10.  
  11. }); 

雖然向前跨了一大步,但好像無濟于事。我還是需要編寫大量的代碼實現(xiàn)此次需求。設計匿名內(nèi)部類的目的,就是為了方便 Java 程序員將代碼作為數(shù)據(jù)傳遞。有時候,匿名內(nèi)部類看這比較復雜,這時候,我突然想起來老大讓我學的lambda表達式,我可以用它來簡化:

  1. List<Melon> europeansLambda = Filters.filterMelons( 
  2.   melons, m -> "europe".equalsIgnoreCase(m.getOrigin()) 
  3. ); 

果然看這帥多了!!!,就這樣,我又一次成功完成了任務。我興沖沖的拿著代碼讓老大去看看。

第六次 引入泛型

老大看著我的代碼說,嗯,不錯!腦袋終于開竅了?,F(xiàn)在你考慮一下,我們的平臺是做農(nóng)產(chǎn)品的,也就是肯定不止瓜這一類水果,如果換做其他的水果,你的代碼如何修改?

目前我們的MelonPredicate僅支持 Melon 類。這家伙怎么搞?說不定哪天他要買蔬菜、海參可怎么搞,總不能給他再創(chuàng)建好多類似MelonPredicate的接口吧。這個時候突然想起老師講過的泛型,該它發(fā)揮作用了!

于是我定義了一個新接口Predicate:

  1. @FunctionalInterface 
  2. public interface Predicate<T> { 
  3.  
  4.   boolean test(T t); 
  5.  

接下來,我們重寫該 filterMelons() 方法并將其重命名為 filter() :

  1. public static <T> List<T> filter(List<T> list, Predicate<T> predicate) { 
  2.    
  3.  List<T> result = new ArrayList<>(); 
  4.  
  5.     for (T t: list) { 
  6.  
  7.       if (t != null && predicate.test(t)) { 
  8.  
  9.         result.add(t); 
  10.  
  11.       } 
  12.  
  13.     }   
  14.  
  15.     return result; 
  16.  

現(xiàn)在,我們可以這樣過濾瓜類 :

  1. List<Melon> watermelons = Filters.filter( 
  2.  
  3.   melons, (Melon m) -> "Watermelon".equalsIgnoreCase(m.getType())); 

同樣的,我們也可以對數(shù)字做同樣的事情:

  1. List<Integer> numbers = Arrays.asList(1, 13, 15, 2, 67); 
  2.  
  3. List<Integer> smallThan10 = Filters.filter(numbers, (Integer i) -> i < 10); 

回過頭來復盤一下,我們發(fā)現(xiàn)自從使用Java 8函數(shù)式接口和lambda表達式后,代碼發(fā)生了明顯的變化。

不知道細心的伙伴有沒有發(fā)現(xiàn)我們上面的 Predicate 接口上面多了一個@FunctionalInterface 上的注解,它就是標記函數(shù)式接口的。

至此,我們通過一個需求的演變過程,了解了lambda和函數(shù)式接口的概念,同時也加深對它們的理解。其實熟悉java8的朋友都知道,在我們的 java.util.function 包下包含40多個此類接口。

函數(shù)式接口和lambda表達式組成了一個強大的團隊。根據(jù)上面的例子,我們知道函數(shù)式接口是我們行為的高度抽象,lambda表達式我們可以看出這種行為的具體實現(xiàn)的一個實例。

  1. Predicate<Melon> predicate = (Melon m)-> "Watermelon".equalsIgnoreCase(m.getType()); 

簡而言之Lambda

lambda表達式由三部分組成,如下圖所示:

以下是lambda表達式各部分的描述:

  • 在箭頭的左側,是在lambda表達式主體中使用的參數(shù)。
  • 在箭頭的右側,是lambda主體 。
  • 箭頭只是lambda參數(shù)和主體的分隔符。

此lambda的匿名類版本如下:

  1. List<Melon> europeans = Filters.filterMelons(melons, new Predicate<Melon>() { 
  2.  
  3.  @Override 
  4.  
  5.  public boolean test(Melon melon) { 
  6.  
  7.    return "Watermelon".equalsIgnoreCase(melon.getType()); 
  8.  
  9.  } 
  10.  
  11. }); 

現(xiàn)在,如果我們查看lambda表達式及其匿名類版本,可以從下面四方面來描述lambda表達式:

我們可以將 lambda 表達式定義為一種 簡潔、可傳遞的匿名函數(shù),首先我們需要明確 lambda 表達式本質(zhì)上是一個函數(shù),雖然它不屬于某個特定的類,但具備參數(shù)列表、函數(shù)主體、返回類型,甚至能夠拋出異常;其次它是匿名的,lambda 表達式?jīng)]有具體的函數(shù)名稱;lambda 表達式可以像參數(shù)一樣進行傳遞,從而簡化代碼的編寫。

Lambda支持行為參數(shù)化,在前面的例子中,我們已經(jīng)證明這一點。最后,請記住,lambda表達式只能在函數(shù)式接口的上下文中使用。

總結

在本文中,我們重點介紹了函數(shù)式接口的用途和可用性,我們將研究如何將代碼從開始的樣板代碼現(xiàn)演變?yōu)榛诤瘮?shù)式接口的靈活實現(xiàn)。希望對大家理解函數(shù)式接口有所幫助,謝謝大家。 

 

責任編輯:武曉燕 來源: 今日Java
相關推薦

2011-11-08 12:37:49

2020-05-02 15:10:53

AI 王者榮耀人工智能

2025-03-24 08:00:00

數(shù)據(jù)庫開發(fā)代碼

2014-07-08 09:21:10

死代碼創(chuàng)意歌曲

2020-09-09 07:13:05

RPC框架

2020-05-29 08:14:49

代碼Try-Catch程序員

2021-07-26 23:39:20

Java變量代碼

2024-11-11 14:57:56

JWTSession微服務

2020-02-07 08:00:29

代碼Java8Bug

2020-11-24 06:17:57

微信代碼移動應用

2022-06-21 12:27:12

JavaScript前端

2020-12-31 10:14:42

防注入代碼繞過

2018-06-19 08:02:00

統(tǒng)計程序微信

2024-11-19 08:36:16

2020-08-13 18:54:53

Python代碼解釋器

2019-10-18 09:39:44

爬蟲消息大數(shù)據(jù)

2019-10-14 09:51:08

爬蟲網(wǎng)絡系統(tǒng)

2013-06-20 11:11:00

程序員經(jīng)理

2024-11-21 09:55:25

2021-02-04 07:55:28

代碼離職互聯(lián)網(wǎng)
點贊
收藏

51CTO技術棧公眾號