舉例解析Java的設(shè)計模式編程中里氏替換原則的意義
里氏替換原則,OCP作為OO的高層原則,主張使用“抽象(Abstraction)”和“多態(tài)(Polymorphism)”將設(shè)計中的靜態(tài)結(jié)構(gòu)改為動態(tài)結(jié)構(gòu),維持設(shè)計的封閉性?!俺橄蟆笔钦Z言提供的功能。“多態(tài)”由繼承語義實現(xiàn)。
里氏替換原則包含以下4層含義:
- 子類可以實現(xiàn)父類的抽象方法,但是不能覆蓋父類的非抽象方法。
- 子類中可以增加自己特有的方法。
- 當子類覆蓋或?qū)崿F(xiàn)父類的方法時,方法的前置條件(即方法的形參)要比父類方法的輸入?yún)?shù)更寬松。
- 當子類的方法實現(xiàn)父類的抽象方法時,方法的后置條件(即方法的返回值)要比父類更嚴格。
現(xiàn)在我們可以對以上四層含義進行講解。
子類可以實現(xiàn)父類的抽象方法,但是不能覆蓋父類的非抽象方法
在我們做系統(tǒng)設(shè)計時,經(jīng)常會設(shè)計接口或抽象類,然后由子類來實現(xiàn)抽象方法,這里使用的其實就是里氏替換原則。子類可以實現(xiàn)父類的抽象方法很好理解,事實上,子類也必須完全實現(xiàn)父類的抽象方法,哪怕寫一個空方法,否則會編譯報錯。
里氏替換原則的關(guān)鍵點在于不能覆蓋父類的非抽象方法。父類中凡是已經(jīng)實現(xiàn)好的方法,實際上是在設(shè)定一系列的規(guī)范和契約,雖然它不強制要求所有的子類必須遵從這些規(guī)范,但是如果子類對這些非抽象方法任意修改,就會對整個繼承體系造成破壞。而里氏替換原則就是表達了這一層含義。
在面向?qū)ο蟮脑O(shè)計思想中,繼承這一特性為系統(tǒng)的設(shè)計帶來了極大的便利性,但是由之而來的也潛在著一些風險。下面舉例來說明繼承的風險,我們需要完成一個兩數(shù)相減的功能,由類A來負責。
class A{
public int func1(int a, int b){
return a-b;
}
}
public class Client{
public static void main(String[] args){
A a = new A();
System.out.println("100-50="+a.func1(100, 50));
System.out.println("100-80="+a.func1(100, 80));
}
}
運行結(jié)果:
100-50=50 100-80=20
后來,我們需要增加一個新的功能:完成兩數(shù)相加,然后再與100求和,由類B來負責。即類B需要完成兩個功能:
兩數(shù)相減。
兩數(shù)相加,然后再加100。
由于類A已經(jīng)實現(xiàn)了第一個功能,所以類B繼承類A后,只需要再完成第二個功能就可以了,代碼如下:
class B extends A{
public int func1(int a, int b){
return a+b;
}
public int func2(int a, int b){
return func1(a,b)+100;
}
}
public class Client{
public static void main(String[] args){
B b = new B();
System.out.println("100-50="+b.func1(100, 50));
System.out.println("100-80="+b.func1(100, 80));
System.out.println("100+20+100="+b.func2(100, 20));
}
}
類B完成后,運行結(jié)果:
100-50=150 100-80=180 100+20+100=220
我們發(fā)現(xiàn)原本運行正常的相減功能發(fā)生了錯誤。原因就是類B在給方法起名時無意中重寫了父類的方法,造成所有運行相減功能的代碼全部調(diào)用了類B重寫后的方法,造成原本運行正常的功能出現(xiàn)了錯誤。在本例中,引用基類A完成的功能,換成子類B之后,發(fā)生了異常。在實際編程中,我們常常會通過重寫父類的方法來完成新的功能,這樣寫起來雖然簡單,但是整個繼承體系的可復用性會比較差,特別是運用多態(tài)比較頻繁時,程序運行出錯的幾率非常大。如果非要重寫父類的方法,比較通用的做法是:原來的父類和子類都繼承一個更通俗的基類,原有的繼承關(guān)系去掉,采用依賴、聚合,組合等關(guān)系代替。
相關(guān)文章
使用easyexcel導出的excel文件,使用poi讀取時異常處理方案
這篇文章主要介紹了使用easyexcel導出的excel文件,使用poi讀取時異常處理方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-12-12
一篇文章帶你了解Java容器,面板及四大布局管理器應(yīng)用
這篇文章主要介紹了JAVA布局管理器與面板組合代碼實例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2021-08-08
MyBatis實現(xiàn)批量插入數(shù)據(jù),多重forEach循環(huán)
這篇文章主要介紹了MyBatis實現(xiàn)批量插入數(shù)據(jù),多重forEach循環(huán)方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-02-02
使用Java實現(xiàn)MapReduce詞頻統(tǒng)計示例代碼
這篇文章主要介紹了使用Java實現(xiàn)MapReduce詞頻統(tǒng)計的相關(guān)資料,通過詞頻統(tǒng)計示例來展示MapReduce的運行機制,涵蓋了Mapper和Reducer的實現(xiàn),并說明了如何配置和執(zhí)行MapReduce作業(yè),需要的朋友可以參考下2024-11-11
Spring Boot 與 kotlin 使用Thymeleaf模板引擎渲染web視圖的方法
這篇文章主要介紹了Spring Boot 與 kotlin 使用Thymeleaf模板引擎渲染web視圖的方法,本文給大家介紹的非常詳細,具有參考借鑒價值,需要的朋友可以參考下2018-01-01
IDEA2020.2.3 "reading maven projects"卡住的問題
這篇文章主要介紹了IDEA2020.2.3 "reading maven projects"卡住的問題及問題原因探究,通過多種方法給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友參考下吧2020-10-10
SpringBoot整合mybatis使用Druid做連接池的方式
這篇文章主要介紹了SpringBoot整合mybatis使用Druid做連接池的方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-08-08

