原來Java接口多實現還可以這樣玩
前言
多繼承的目的是為瞭提高子類的功能,擴展性得到瞭提高。
為瞭擴充子類的功能,java改良成多實現。這裡帶來兩個問題,多實現就沒有不確定性問題瞭麼?接口中都是抽象方法,實現多實現結構還要重寫,意義大麼?
第二個問題,就是在於要重寫方法,而多實現解決的是讓實現的子類具備一些功能,至於重寫函數的麻煩不在考慮的范圍之內。那麼第一個不確定性問題呢?
多繼承不支持,是因為方法的不確定性,聲明相同,方法體不同,子類在繼承執行相同的方法時,不知道執行哪個方法體。毛病出現在方法體上。禁止使用多繼承,就是為瞭預防出現這樣的問題。但是接口中定義的全是抽象方法,在多個繼承的時候(說法不正確,要說成實現,但是本質還是繼承覆蓋),即使碰到相同的函數,也沒關系,因為裡面沒有方法體,要執行的方法體是子類的自己重寫,壓根不會產生要執行多個不同的方法體的選擇問題。而且,我認為多接口中,根本不會出現相同的方法,因為,編寫的時候是從上往下的書寫的。隻是在最初在理解的時候,是從下往上的,可能要遇到這些問題。多繼承不執行,也有父類中有一般函數,不是抽象函數的問題。不是說子類多繼承幾個父類不行,有一些情況也是可以的,但是為瞭預防出現的問題,所以禁止它瞭。
接口是給子類提高瞭一些功能,告訴子類你可以具備哪些功能,至於怎麼實現這些功能,需要子類自己書寫。
Java中的接口類通常是為瞭提取共同點,規范實現,便於閱讀,處理好接口類多實現並提供優雅的命中具體實現,能夠幫助我們簡化代碼,提高可讀性;下面介紹幾種用起來很舒服的多實現方式及調用方式供大傢參考。
示例
枚舉實現
接口定義
public interface Breakfast { void eat(); }
實現
public enum BreakfastEnum implements Breakfast { Beijing("北京") { @Override public void eat() { System.out.println("北京人早餐吃豆汁和焦圈"); } }, Wuhan("武漢") { @Override public void eat() { System.out.println("武漢人早餐吃熱幹面、豆皮..."); } }, Unknown("未知") { @Override public void eat() { System.out.println("不吃早餐!"); } }; private String city; BreakfastEnum(String city) { this.city = city; } private String getCity() { return this.city; } /** * 提供統一入口 找到對應子類並執行 * * @param city */ public static void eat(String city) { BreakfastEnum[] values = BreakfastEnum.values(); Arrays.stream(values).filter(e -> city.equals(e.city)).findFirst().orElse(Unknown).eat(); } }
測試一下
通過枚舉類實現接口,每一個枚舉相當於一個實現,在代碼塊實現方法即可,最後在枚舉類提供一個靜態方法作為統一入口,調用方便,代碼簡潔,提供通用實現處理無特定實現的場景,適合用於替換ifelse較多的業務代碼,優化復雜的工具類等等,對於方法很多,業務復雜的業務慎用。
常規多實現(調用示例)
業務場景
我們有一個消息服務用於監聽消息並發送到客戶端,消息中有一個發佈方式字段
1.根據消息的發佈方式字段發送到指定途徑
2.將消息發到所有途徑
接口定義
public interface MessageHandle { /** * 發佈消息 * * @param msg */ void publish(JSONObject msg); }
實現
/** * 發送短信 */ @Service("sms") public class SmsMessageHandle implements MessageHandle { @Override public void publish(JSONObject msg) { // 發送短信 // 省略實現... } } /** * 推送 */ @Service("push") public class PushMessageHandle implements MessageHandle { @Override public void publish(JSONObject msg) { // 推送到app // 省略實現... } }
應用
// 1.指定途徑發送 @Component public class MessageListener { @Autowired private Map<String, MessageHandle> messageHandleMap; @KafkaListener(groupId = "message-server", topics = "message") public void listener(String message, Acknowledgment ack) { JSONObject messageJson = JSON.parseObject(message); // 獲取發佈方式 sms push...對應實現@Service註解中的名稱 String publishType = messageJson.getString("publishType"); // 獲取實現 MessageHandle handle = messageHandleMap.get(publishType); if(handle != null) { handle.publish(messageJson); } // 提交偏移量 ack.acknowledge(); } }
// 2.每種途徑都發送 @Component public class MessageListener { @Autowired private List<MessageHandle> messageHandleList; @KafkaListener(groupId = "message-server", topics = "message") public void listener(String message, Acknowledgment ack) { JSONObject messageJson = JSON.parseObject(message); // 每種途徑都發送 for(MessageHandle handle : messageHandleList){ handle.publish(messageJson); } // 提交偏移量 ack.acknowledge(); } }
總結
到此這篇關於Java接口多實現的文章就介紹到這瞭,更多相關Java接口多實現內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!