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

Java8新特性系列-默認(rèn)方法和靜態(tài)方法

開(kāi)發(fā) 前端
在Java 8之前,默認(rèn)情況下,接口中的所有方法都是公共的和抽象的。但是這一限制在Java 8中被打破了,Java 8允許開(kāi)發(fā)人員在接口中添加新方法,而無(wú)需在實(shí)現(xiàn)這些接口的類(lèi)中進(jìn)行任何更改。

[[385113]]

 在Java 8之前,默認(rèn)情況下,接口中的所有方法都是公共的和抽象的。但是這一限制在Java 8中被打破了,Java 8允許開(kāi)發(fā)人員在接口中添加新方法,而無(wú)需在實(shí)現(xiàn)這些接口的類(lèi)中進(jìn)行任何更改。

為什么會(huì)有默認(rèn)方法?

主要是為了方便擴(kuò)展已有接口;如果沒(méi)有默認(rèn)方法,假如給Java中的某個(gè)接口添加一個(gè)新的抽象方法,那么所有實(shí)現(xiàn)了該接口的類(lèi)都得修改,影響將非常大。

舉個(gè)例子,Sortable 接口以及實(shí)現(xiàn)該接口的類(lèi)SortableNumberCollection和SortableStringCollection。該接口有兩種方法:void sort(); 和T peek()。

  1. public interface Sortable<T> { 
  2.     void sort(); 
  3.     T peek(); 

 sort()方法用于對(duì)象排序,T peek()用于獲取指定元素,另外需要一個(gè)比較器類(lèi)ObjectComparator來(lái)對(duì)對(duì)象進(jìn)行排序。

  1. public class ObjectComparator implements Comparator<Comparable> { 
  2.     @Override 
  3.     public int compare(Comparable o1, Comparable o2) { 
  4.         return o1.compareTo(o2); 
  5.     } 

 SortableStringCollection是一個(gè)自定義集合類(lèi)可以進(jìn)行排序,并查看字符串指定元素,代碼如下:

  1. public class SortableStringCollection implements Sortable<String> { 
  2.  
  3.     private List<String> items = new ArrayList<>(); 
  4.  
  5.     public void add(String item) { 
  6.         items.add(item); 
  7.     } 
  8.  
  9.     @Override 
  10.     public void sort() { 
  11.         items.sort(new ObjectComparator()); 
  12.     } 
  13.  
  14.     @Override 
  15.     public String peek() { 
  16.         return items.get(0); 
  17.     } 

 同樣,SortableNumberCollection是一個(gè)自定義集合類(lèi),其中包含可以使用接口方法進(jìn)行排序和查看的數(shù)字列表指定元素,代碼如下:

  1. public class SortableNumberCollection implements Sortable<Integer> { 
  2.  
  3.     private List<Integer> items = new ArrayList<>(); 
  4.  
  5.     public void add(Integer item) { 
  6.         items.add(item); 
  7.     } 
  8.  
  9.     @Override 
  10.     public void sort() { 
  11.         items.sort(new ObjectComparator()); 
  12.     } 
  13.  
  14.     @Override 
  15.     public Integer peek() { 
  16.         return items.get(0); 
  17.     } 

 在Java8之前如果對(duì)接口Sortable添加新方法:T sortAndPeek(),那么SortableStringCollection和

SortableNumberCollection都必須實(shí)現(xiàn)T sortAndPeek()方法。

Java8之后提供了一種新的實(shí)現(xiàn)方式,默認(rèn)方法 default method,我們可以對(duì)Sortable進(jìn)行如下改造:

  1. public interface Sortable<T> { 
  2.     void sort(); 
  3.     T peek(); 
  4.  
  5.     default T sortAndPeek(){    // New 'default method' added in the interface 
  6.         sort(); 
  7.         return peek(); 
  8.     } 
  9.  

 同時(shí)SortableStringCollection和SortableNumberCollection類(lèi)不需要任何更改。這樣可以減少我們對(duì)原有代碼的改動(dòng)。同時(shí)如果需要,還可以在實(shí)現(xiàn)此接口的任何類(lèi)中重寫(xiě)該方法T sortAndPeek()的默認(rèn)實(shí)現(xiàn)。

在下圖中我們看到default Method不通的標(biāo)識(shí):


在多繼承中使用默認(rèn)方法問(wèn)題

如果兩個(gè)或多個(gè)接口具有相同的默認(rèn)方法簽名,并且一個(gè)類(lèi)實(shí)現(xiàn)了這兩個(gè)接口,則將引發(fā)編譯時(shí)錯(cuò)誤。例如:

  1. public interface Interface1 { 
  2.     void methodOne(String str); 
  3.     default void newMethod(){ 
  4.         System.out.println("Interface1: Newly added method"); 
  5.     } 
  6.  
  7.  
  8.  
  9. public interface Interface2 { 
  10.     void methodTwo(String str); 
  11.     default void newMethod(){ 
  12.         System.out.println("Interface2: Newly added method"); 
  13.     } 
  14.  
  15.  
  16.  
  17. public class InterfaceImplementation implements Interface1, Interface2{ 
  18.     @Override 
  19.     public void methodOne(String str) { 
  20.         System.out.println("Overridden methodOne: " + str); 
  21.     } 
  22.  
  23.     @Override 
  24.     public void methodTwo(String str) { 
  25.         System.out.println("Overridden methodTwo: " + str ); 
  26.     } 

 此時(shí)代碼會(huì)提示如下異常:

  1. InterfaceImplementation inherits unrelated defaults for newMethod() from types Interface1 and Interface2 

要解決此問(wèn)題,我們將必須重寫(xiě)類(lèi)InterfaceImplementation中的方法:

  1. public class InterfaceImplementation implements Interface1, Interface2{ 
  2.     @Override 
  3.     public void methodOne(String str) { 
  4.         System.out.println("Overridden methodOne: " + str); 
  5.     } 
  6.  
  7.     // newMethod implemented to resolve the conflict. 
  8.     @Override 
  9.     public void newMethod() { 
  10.         System.out.println("InterfaceImplementation: Newly added method"); 
  11.     } 
  12.  
  13.     @Override 
  14.     public void methodTwo(String str) { 
  15.         System.out.println("Overridden methodTwo: " + str ); 
  16.     } 

 我們總結(jié)一下:

  • 類(lèi)中的方法優(yōu)先級(jí)最高。類(lèi)或父類(lèi)中聲明的方法的優(yōu)先級(jí)高于任何聲明為默認(rèn)方法的優(yōu)先級(jí)。
  • 如果無(wú)法依據(jù)第一條進(jìn)行判斷,那么子接口的優(yōu)先級(jí)更高:函數(shù)簽名相同時(shí),優(yōu)先選擇擁有最具體實(shí)現(xiàn)的默認(rèn)方法的接口,即如果B繼承了A,那么B就比A更加具體。
  • 最后,如果還是無(wú)法判斷,繼承了多個(gè)接口的類(lèi)必須通過(guò)顯式覆蓋和調(diào)用期望的方法,顯式地選擇使用哪一個(gè)默認(rèn)方法的實(shí)現(xiàn)。

在Java 8中添加靜態(tài)方法

接口定義的靜態(tài)方法獨(dú)立于任何對(duì)象調(diào)用。所以,在調(diào)用靜態(tài)方法時(shí),不需要實(shí)現(xiàn)接口,也不需要接口的實(shí)例,

就像“默認(rèn)方法”一樣,“靜態(tài)方法”也可以添加到接口中。例如,我們可以添加一個(gè)靜態(tài)方法Direction getDefaultDirection(),該方法將返回默認(rèn)Direction,例如:

  1. public interface Sortable<T> { 
  2.  
  3.     Direction defaultDirection = Direction.DESC
  4.  
  5.     enum Direction { 
  6.         ASC
  7.         DESC 
  8.     }; 
  9.  
  10.     void sort(); 
  11.     T peek(); 
  12.  
  13.     static Direction getDefaultDirection(){    // 'static method' added to the interface. 
  14.         return defaultDirection; 
  15.     } 

 在上面的示例中,可以使用類(lèi)引用來(lái)調(diào)用靜態(tài)Direction getDefaultDirection()方法:

  1. Sortable.getDefaultDirection() 

對(duì)默認(rèn)方法和靜態(tài)方法的一點(diǎn)思考

接口是設(shè)計(jì)模式中一種開(kāi)閉原則的體驗(yàn),而java8賦予了接口新的特性,使得接口使用起來(lái)更加的得心應(yīng)手了,這也有助于我們更加內(nèi)聚自己的代碼結(jié)構(gòu)了。Java源碼中也有很多場(chǎng)景使用到了默認(rèn)方法,例如:Iterator接口,我們?cè)陂_(kāi)發(fā)中可以多使用一些新的特性從而提高開(kāi)發(fā)效率及增加代碼的健壯性。

  1. public interface Iterable<T> { 
  2.  
  3.     Iterator<T> iterator(); 
  4.      
  5.     default void forEach(Consumer<? super T> action) { 
  6.         Objects.requireNonNull(action); 
  7.         for (T t : this) { 
  8.             action.accept(t); 
  9.         } 
  10.     } 
  11.      
  12.     default Spliterator<T> spliterator() { 
  13.         return Spliterators.spliteratorUnknownSize(iterator(), 0); 
  14.     } 
  15.      

 【編輯推薦】

 

責(zé)任編輯:姜華 來(lái)源: 今日頭條
相關(guān)推薦

2021-09-13 07:53:30

安全

2014-04-10 18:00:10

Java8Java8教程

2021-03-02 07:13:54

Java8版本升級(jí)

2021-02-22 11:51:15

Java開(kāi)發(fā)代碼

2023-07-19 08:03:05

Future異步JDK

2024-10-09 08:42:03

2022-12-09 07:48:10

Java8Stream表達(dá)式

2022-12-30 09:24:23

Java8Stream操作

2014-07-15 14:48:26

Java8

2009-08-27 16:24:48

擴(kuò)展方法C# 3.0新特性

2014-03-28 11:08:16

Java 8多重繼承

2014-10-20 13:57:59

JavaFX 8Java 8

2015-05-25 09:45:16

Java多繼承深入解析

2017-10-25 11:05:14

Java

2014-07-14 11:34:53

Java 8Nashorn

2020-12-10 10:11:52

Javastatic關(guān)鍵字

2014-04-11 12:49:00

Java8Java8教程

2025-02-11 08:31:37

Java關(guān)鍵字安全

2014-04-15 15:45:22

Java8Java8教程

2014-03-19 11:04:14

Java 8Java8特性
點(diǎn)贊
收藏

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