有時候一個對象的方法可能不是我們想要的功能,我們希望能將這個方法覆寫。而對於覆寫,我們最直白的感覺就是通過子類繼承的方式,但是有時候對於使用web開發而言,我們能知道獲取對象的實現接口,而真正對象是屬於哪個類我們並不知道,例如Connection接口,HttpServletResponse接口這樣的,我們並不知道第三方Jar包或者別的框架具體的實現類,那么我們只能通過已知的接口方式來增強或改寫某個對象的方法。
通常增強某個對象方法的方式有三種:
⑴ 使用子類繼承某個實現類;
⑵ 使用包裝設計模式;
⑶ 使用動態代理。
本篇主要講解包裝設計模式,但要知道動態代理是最佳的解決方法。
包裝設計模式主要有如下五個步驟:
① 自定義一個類,實現與目標對象(被增強對象)相同的接口。
② 在自定義的類中定義一個成員變量,來記住(指向)目標對象。
③ 定義一個構造函數,用來接收目標對象。
④ 覆蓋想增強或改寫的方法。
⑤ 對於其他不想增強或改寫的方法,使用剛才記住目標對象的成員變量來一一調用目標對象的方法即可。
以數據庫連接的Connection接口為例,我們在數據庫連接池中給其他方法的Connection對象必須不能是數據庫直接提供的Connection對象,因為其close方法會將連接直接銷毀,而我們應該返回的是調用close方法能將連接重新放回池子中的Connection對象,所以我們使用一個包裝類來封裝數據庫提供的Connection對象,覆寫其close方法,對於其他方法,則直接使用原來的對象去調用,代碼如下:
1 class MyConnection implements Connection{ //包裝設計模式的類 2 private Connection conn; 3 public MyConnection(Connection conn) { //接收目標對象 4 this.conn = conn; 5 } 6 7 @Override 8 public void close() throws SQLException { 9 connectionList.addFirst(this.conn); //調用close方法時只是將連接重新返回池中,而不會銷毀 10 } 11 12 @Override 13 public Statement createStatement() throws SQLException { 14 this.conn.createStatement(); //對於不增強的方法則調用目標對象的方法即可,在MyConnection包裝類中其他方法都是這樣的 15 return null; 16 } 17 。。。 //以下省略Connection接口中覆寫的其他方法,對於不想增強的方法都如上(createStatement方法)所示 18 } // MyConnection包裝類完成