3atv精品不卡视频,97人人超碰国产精品最新,中文字幕av一区二区三区人妻少妇,久久久精品波多野结衣,日韩一区二区三区精品

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 运维知识 > Android >内容正文

Android

Android架构演进 · 设计模式· 为什么建议你一定要学透设计模式?

發(fā)布時(shí)間:2023/12/29 Android 42 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Android架构演进 · 设计模式· 为什么建议你一定要学透设计模式? 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

一、引言

Hello,我是小木箱,歡迎來(lái)到小木箱成長(zhǎng)營(yíng)Android架構(gòu)演進(jìn)系列教程,今天將分享Android架構(gòu)演進(jìn) · 設(shè)計(jì)模式· 為什么建議你一定要學(xué)透設(shè)計(jì)模式?

今天分享的內(nèi)容主要分為四部分內(nèi)容,第一部分是設(shè)計(jì)模式5W2H,第二部分是7大設(shè)計(jì)原則,第三部分是3大設(shè)計(jì)模式,最后一部分是總結(jié)與展望。

其中,7大設(shè)計(jì)原則主要包括開閉原則、里氏替換原則、依賴倒置原則、單一職責(zé)原則、接口隔離原則、最小知識(shí)原則和合成復(fù)用原則。3大設(shè)計(jì)模式主要包括創(chuàng)建型、結(jié)構(gòu)型和行為型。

拿破侖之前說(shuō)過(guò)Every French soldier carries a marshal’s baton in his knapsack,意思是“每個(gè)士兵背包里都應(yīng)該裝有元帥的權(quán)杖”,本意是激勵(lì)每一名上戰(zhàn)場(chǎng)的士兵都要有大局觀,有元帥的思維。

編程的海洋里也是如此,不想當(dāng)架構(gòu)師的程序員不是好程序員,我覺得架構(gòu)師可能是大部分程序員最理想的歸宿。而設(shè)計(jì)模式是每一個(gè)架構(gòu)師所必備的技能之一,只有學(xué)透了設(shè)計(jì)模式,才敢說(shuō)真正理解了軟件工程。

希望每個(gè)有技術(shù)追求的Android開發(fā),可以從代碼中去尋找屬于自己的那份快樂(lè),通過(guò)代碼構(gòu)建編程生涯的架構(gòu)思維。

如果學(xué)完小木箱Android架構(gòu)演進(jìn)設(shè)計(jì)模式系列文章,那么任何人都能為社區(qū)或企業(yè)貢獻(xiàn)優(yōu)質(zhì)的SDK設(shè)計(jì)方案。

二、設(shè)計(jì)模式5W2H

5W2H又叫七何分析法,5W2H是二戰(zhàn)的時(shí)候,美國(guó)陸軍兵器修理部發(fā)明的思維方法論,便于啟發(fā)性的理解深水區(qū)知識(shí)。5W2H是What、Who、Why、Where、When、How much、How首字母縮寫之和,廣泛用于企業(yè)技術(shù)管理、頭腦風(fēng)暴等。小木箱今天嘗試用5W2H分析法分析一定要學(xué)透設(shè)計(jì)模式底層邏輯。

2.1 What: 什么是設(shè)計(jì)模式?

首先聊聊設(shè)計(jì)模式5W2H的What, 什么是設(shè)計(jì)模式?sourcemaking曾經(jīng)提過(guò),在軟件工程中,設(shè)計(jì)模式是軟件設(shè)計(jì)中,常見問(wèn)題的可重復(fù)解決方案。設(shè)計(jì)模式雖然不可以直接轉(zhuǎn)換為代碼,然后完成設(shè)計(jì)。但是設(shè)計(jì)模式是解決不同情況特定共性問(wèn)題的通用模板。

不管北京的四合院、廣州的小蠻腰還是上海的東方明珠,都有好的建筑地基。設(shè)計(jì)模式也是如此,設(shè)計(jì)模式是高效能工程建設(shè)基石,如果業(yè)務(wù)代碼看作鋼筋水泥,那么設(shè)計(jì)模式可以看作是建筑地基。只有地基足夠牢固,項(xiàng)目工程才不會(huì)因?yàn)橘|(zhì)量問(wèn)題爛尾。

如果想高度重用代碼,那么建議你一定要學(xué)透設(shè)計(jì)模式。

如果想讓代碼更容易被理解,那么建議你一定要學(xué)透設(shè)計(jì)模式。

如果想確保代碼可靠性、可維護(hù)性,那么建議你一定要學(xué)透設(shè)計(jì)模式。

簡(jiǎn)而言之,設(shè)計(jì)模式不僅是被多數(shù)人知曉、被反復(fù)驗(yàn)證的代碼設(shè)計(jì)經(jīng)驗(yàn)總結(jié),而且設(shè)計(jì)模式是特定場(chǎng)景和業(yè)務(wù)痛點(diǎn),針對(duì)同類問(wèn)題通用解決方案。

2.2 Who: 誰(shuí)應(yīng)該學(xué)透設(shè)計(jì)模式?

聊完設(shè)計(jì)模式5W2H的What,再聊聊設(shè)計(jì)模式5W2H的Who,誰(shuí)應(yīng)該學(xué)透設(shè)計(jì)模式? 設(shè)計(jì)模式可以用于所有項(xiàng)目開發(fā)工作的一種高價(jià)值設(shè)計(jì)理念,而且在寫源代碼或者閱讀源代碼中經(jīng)常需要用到。

如果你的工作是偏架構(gòu)方向,那么使用設(shè)計(jì)模式可能像一日三餐一樣頻繁。設(shè)計(jì)模式既然鏈接著每個(gè)基礎(chǔ)模塊的方方面面,就看你想不想讓你的編程生涯走的更遠(yuǎn),如果想,就接著往下面看。

設(shè)計(jì)模式不是代碼! 設(shè)計(jì)模式不是代碼! 設(shè)計(jì)模式不是代碼!?重要的事情說(shuō)三遍,設(shè)計(jì)模式是編程思想。如果參加外企面試,那么基本不會(huì)像國(guó)內(nèi)考各種八股文,有兩個(gè)重點(diǎn)考查項(xiàng)目,一方面是算法,另一方面是系統(tǒng)設(shè)計(jì)。而系統(tǒng)設(shè)計(jì)和設(shè)計(jì)模式息息相關(guān)。

如果面試國(guó)內(nèi)中大廠架構(gòu)組也是,設(shè)計(jì)模式對(duì)于架構(gòu)組程序員而言,基本隨便拿捏,設(shè)計(jì)模式是區(qū)分中級(jí)程序員和高級(jí)程序員的關(guān)鍵。當(dāng)國(guó)內(nèi)Android程序員疲于業(yè)務(wù),導(dǎo)致設(shè)計(jì)模式在業(yè)務(wù)方面缺少實(shí)踐,如果你掌握的比他們更好,是不是相比之下會(huì)更有競(jìng)爭(zhēng)力。學(xué)透設(shè)計(jì)模式是普通開發(fā)逆襲架構(gòu)師的捷徑,但凡有一定工作年限的Android開發(fā),都必須學(xué)透設(shè)計(jì)模式

2.3 Why: 為什么要學(xué)透設(shè)計(jì)模式?

聊完設(shè)計(jì)模式5W2H的Who,再聊聊設(shè)計(jì)模式5W2H的Why,為什么要學(xué)透設(shè)計(jì)模式? 關(guān)于為什么要學(xué)透設(shè)計(jì)模式的原因一共有三點(diǎn)。

第一,學(xué)透設(shè)計(jì)模式,為職場(chǎng)晉升打開綠色通道。?代碼是程序員的一個(gè)門面。有些互聯(lián)網(wǎng)大廠技術(shù)崗晉升,會(huì)隨機(jī)抽取對(duì)方代碼提交節(jié)點(diǎn),根據(jù)對(duì)方的代碼片段,進(jìn)行review并給予晉升打分。這是平時(shí)為什么要注重代碼整潔性很重要原因。學(xué)透了設(shè)計(jì)模式并應(yīng)用于項(xiàng)目開發(fā),等同你有了職場(chǎng)晉升直升飛機(jī)。

第二,學(xué)透設(shè)計(jì)模式,編碼全局觀更強(qiáng)。?正確使用設(shè)計(jì)模式,無(wú)異于站在巨人的肩膀上看世界。前輩們?cè)?Java 、C++等語(yǔ)言領(lǐng)域上,花費(fèi)幾十年時(shí)間,并經(jīng)過(guò)分類、提煉、總結(jié)、沉淀和驗(yàn)證等各個(gè)方面的努力,才整理了一套精品架構(gòu)思想,如果想成為一名Senior Android Dev,那么有什么理由不去研究呢?

第三,學(xué)透設(shè)計(jì)模式,抽象建模能力更強(qiáng)。?對(duì)于特定場(chǎng)景和特定業(yè)務(wù),如果正確的使用設(shè)計(jì)模式,那么代碼質(zhì)量和業(yè)務(wù)拓展性會(huì)有質(zhì)的飛躍。

2.4 When: 什么時(shí)候使用設(shè)計(jì)模式?

說(shuō)完設(shè)計(jì)模式5W2H的Who,再聊聊設(shè)計(jì)模式5W2H的When,關(guān)于使用設(shè)計(jì)模式的時(shí)機(jī)一共有兩點(diǎn)。

第一點(diǎn),場(chǎng)景要吻合

第二點(diǎn),確保原有業(yè)務(wù)穩(wěn)定基礎(chǔ)上,套用或靈活運(yùn)用設(shè)計(jì)模式,可以解決未來(lái)可能出現(xiàn)的拓展性和維護(hù)性問(wèn)題。

2.5 Where: 哪些地方需要使用到設(shè)計(jì)模式?

說(shuō)完設(shè)計(jì)模式5W2H的When,再聊聊設(shè)計(jì)模式5W2H的Where,哪些地方需要使用到設(shè)計(jì)模式?多數(shù)情況下,如果程序員做技術(shù)需求要考慮其靈活性的地方,就可以使用設(shè)計(jì)模式。22種設(shè)計(jì)模式都有自己的策閱,22種設(shè)計(jì)模式的策閱也適合不同的場(chǎng)景。

我們不可能從策閱設(shè)計(jì)模式的業(yè)務(wù)背景套用狀態(tài)設(shè)計(jì)模式的業(yè)務(wù)背景,好比女朋友,世界上沒(méi)有性格一模一樣的女生,一個(gè)女生只能解決當(dāng)時(shí)狀態(tài)的情感需要。我們的前任和現(xiàn)任帶給的體感都不完全一樣。因此,每一種設(shè)計(jì)模式中的策閱和女朋友一樣,每一個(gè)都是原創(chuàng)

設(shè)計(jì)模式和女朋友有一個(gè)共同特征就是提供了當(dāng)時(shí)背景下可以解決問(wèn)題(情緒價(jià)值)的結(jié)構(gòu)。在解決實(shí)際問(wèn)題時(shí),必須考慮該問(wèn)題解決方案的變動(dòng),如果業(yè)務(wù)發(fā)生大的變動(dòng),那么需要考慮設(shè)計(jì)模式是否通用。好比女朋友,如果當(dāng)下你習(xí)慣內(nèi)卷,但女朋友突然躺平了,那么后面話題可能越來(lái)越少。

使用設(shè)計(jì)模式切勿生搬硬套,正確的使用設(shè)計(jì)模式可以很好地將技術(shù)需求映射業(yè)務(wù)模型上。但是如果過(guò)度使用不合適的設(shè)計(jì)模式會(huì)造成程序可讀性更高,維護(hù)成本變得更高。好比女朋友,如果為了女朋友過(guò)度忍讓,那么最終可能因?yàn)殛P(guān)系不平等不歡而散。

那么,怎樣挑選合適的設(shè)計(jì)模式呢?使用設(shè)計(jì)模式的準(zhǔn)則是在建立對(duì)設(shè)計(jì)模式有很好認(rèn)知前提,并習(xí)慣這種方法模型,看到一種特定的技術(shù)背景,就立馬可以聯(lián)想到具體對(duì)應(yīng)的模型。這種地方使用設(shè)計(jì)模式是最合適不過(guò)的。好比追女生,如果能對(duì)方興趣愛好和性格特征能相互吸引,各方面背景匹配度高,會(huì)更合適一點(diǎn)。

綜上所述,如果你發(fā)現(xiàn)特定業(yè)務(wù)痛點(diǎn),剛好符合特定設(shè)計(jì)原則,或能匹配特定設(shè)計(jì)模式方法模型,那么建議你將這種業(yè)務(wù)抽象成通用模板映射到實(shí)際業(yè)務(wù)里。

2.6 How much: 學(xué)透設(shè)計(jì)模式的價(jià)值點(diǎn)是什么?

說(shuō)完設(shè)計(jì)模式5W2H的Where,再聊聊設(shè)計(jì)模式5W2H的How much,學(xué)透設(shè)計(jì)模式的價(jià)值點(diǎn)是什么?關(guān)于使用設(shè)計(jì)模式的價(jià)值一共有三點(diǎn),第一點(diǎn)是針對(duì)個(gè)人,第二點(diǎn)是針對(duì)工程質(zhì)量,最后一點(diǎn)是針對(duì)團(tuán)隊(duì)。

對(duì)個(gè)人而言,正確使用設(shè)計(jì)模式可以提高程序代碼設(shè)計(jì)能力和職場(chǎng)受歡迎度。

對(duì)工程質(zhì)量而言,如果想要代碼高可用、高復(fù)用、可讀性強(qiáng)和擴(kuò)展性強(qiáng),那么需要設(shè)計(jì)模式做支撐。

對(duì)團(tuán)隊(duì)而言,在現(xiàn)有工業(yè)化和商業(yè)化的代碼設(shè)計(jì)維度上,設(shè)計(jì)模式不僅更標(biāo)準(zhǔn)和更工程化,而且設(shè)計(jì)模式可以提高編碼開發(fā)效率,節(jié)約解決問(wèn)題時(shí)間。

2.7 How: 怎樣學(xué)透設(shè)計(jì)模式?

說(shuō)完設(shè)計(jì)模式5W2H的How much,再聊聊設(shè)計(jì)模式5W2H的How,怎樣學(xué)透設(shè)計(jì)模式?學(xué)透設(shè)計(jì)模式有四種途徑,分別是網(wǎng)課、文章、書籍、源碼和項(xiàng)目實(shí)戰(zhàn)。網(wǎng)課方面,小木箱推薦大家在B站看一下馬士兵教育和圖靈課堂視頻。這兩門課程可以帶大家很輕松的入門設(shè)計(jì)模式。

文章方面,小木箱推薦大家看一下百度工程師教你玩轉(zhuǎn)設(shè)計(jì)模式(觀察者模式)、?提升代碼質(zhì)量的方法:領(lǐng)域模型、設(shè)計(jì)原則、設(shè)計(jì)模式、洞察設(shè)計(jì)模式的底層邏輯、設(shè)計(jì)模式二三事、設(shè)計(jì)模式在業(yè)務(wù)系統(tǒng)中的應(yīng)用、Android中竟然包含這么多設(shè)計(jì)模式,一起來(lái)學(xué)一波!、當(dāng)設(shè)計(jì)模式遇上 Hooks、談?wù)勎夜ぷ髦械?3個(gè)設(shè)計(jì)模式和設(shè)計(jì)模式之美等文章。

書籍方面,小木箱推薦大家看一下《head first》?、《重學(xué)Java設(shè)計(jì)模式 RPC中間件設(shè)計(jì)應(yīng)用程序設(shè)計(jì)編程實(shí)戰(zhàn)分布式領(lǐng)域驅(qū)動(dòng)設(shè)計(jì)和設(shè)計(jì)模式結(jié)合》?和?《代碼整潔之道》?。

源碼方面,Glog日志框架可以值得一學(xué)。

項(xiàng)目實(shí)戰(zhàn)方面,學(xué)有余力的同學(xué)可以動(dòng)手用設(shè)計(jì)模式實(shí)現(xiàn)一下定位組件、實(shí)時(shí)日志組件和啟動(dòng)監(jiān)控組件。

最后,聽說(shuō)有個(gè)叫小木箱這個(gè)家伙設(shè)計(jì)模式的文章寫的還挺不錯(cuò)的,可以關(guān)注一下~

三、7大設(shè)計(jì)原則

產(chǎn)生代碼差的原因,有兩方面,第一方面是外部原因,第二方面是內(nèi)部原因。外部原因主要有:項(xiàng)目排期急,沒(méi)有多少時(shí)間去設(shè)計(jì);資源短缺,人手不夠,只能怎么快怎么來(lái);緊急問(wèn)題修復(fù),臨時(shí)方案快速處理……。內(nèi)部原因主要有:自我要求不高;無(wú)反饋通道

而解決代碼差的根因主要是方法有三個(gè):領(lǐng)域建模、設(shè)計(jì)原則、設(shè)計(jì)模式

分析階段:當(dāng)拿到一個(gè)需求時(shí),先不要著急想著怎么把這個(gè)功能實(shí)現(xiàn),這種很容易陷入事務(wù)腳本的模式。

  • 分析什么呢?需要分析需求的目的是什么、完成該功能需要哪些實(shí)體承擔(dān),這一步核心是找實(shí)體。
  • 舉個(gè)上面進(jìn)店Tab展示的例子,它有兩個(gè)關(guān)鍵的實(shí)體:導(dǎo)航欄、Tab,其中導(dǎo)航欄里面包含了若干個(gè)Tab。

    設(shè)計(jì)階段:分析完了有哪些實(shí)體后,再分析職責(zé)如何分配到具體的實(shí)體上,這就要運(yùn)用一些設(shè)計(jì)原則去指導(dǎo)

    回到上面的例子上,Tab的職責(zé)主要有兩個(gè):一個(gè)是Tab能否展示,這是它自己的職責(zé),如上新Tab展示的邏輯是店鋪30天內(nèi)有上架新商品;

    另一個(gè)職責(zé)就是Tab規(guī)格信息的構(gòu)建,也是它自己要負(fù)責(zé)的。

    導(dǎo)航欄的職責(zé)有兩個(gè):一個(gè)是接受Tab注冊(cè);另一個(gè)是展示。職責(zé)分配不合理,也就不滿足高內(nèi)聚、低耦合的特征。

    打磨階段:這個(gè)階段選擇合適的模式去實(shí)現(xiàn),大家一看到模式都會(huì)理解它是做什么的,比如看到模板類,就會(huì)知道處理通用的業(yè)務(wù)流程,具體變化的部分放在子類中處理。

    上面的這個(gè)例子,用到了2個(gè)設(shè)計(jì)模式:一個(gè)是訂閱者模式,Tab自動(dòng)注冊(cè)的過(guò)程;另一個(gè)是模板模式,先判斷Tab能否展示,然后再構(gòu)建Tab規(guī)格信息,流程雖然簡(jiǎn)單,也可以抽象出來(lái)通用的流程出來(lái),子類只用簡(jiǎn)單地重寫2個(gè)方法。

    領(lǐng)域模型主要是和產(chǎn)品和運(yùn)營(yíng)梳理業(yè)務(wù)模型,進(jìn)行流程化優(yōu)化,進(jìn)而判斷需求是否合理可行。

    提升代碼質(zhì)量還有一個(gè)捷徑,那就是要遵循七大原則,七大原則好比毛澤東農(nóng)村包圍城市指導(dǎo)方針。首先確定統(tǒng)一中國(guó)目標(biāo),然后是在統(tǒng)治力量薄弱的農(nóng)村建立革命根據(jù)地,等革命隊(duì)伍變大,建立農(nóng)村包圍城市的矩陣,最后采取不同摧毀策閱對(duì)國(guó)民政府不同城市政權(quán)進(jìn)行各個(gè)擊破。

    如果系統(tǒng)工程業(yè)務(wù)代碼混亂,我們首先確保底層代碼功能不變,然后以點(diǎn)成線,以線成面,以面成網(wǎng),以網(wǎng)建模。根據(jù)設(shè)計(jì)原則,針對(duì)不同的業(yè)務(wù)痛點(diǎn),制定單一原則或組合原則技術(shù)方案。接著小步快跑,穩(wěn)定安全地實(shí)施軟件工程質(zhì)量改造規(guī)劃,最終達(dá)到降低業(yè)務(wù)冗余或者降低未來(lái)大幅度代碼變更帶來(lái)的風(fēng)險(xiǎn)目的。設(shè)計(jì)原則的底層邏輯就是讓軟件能夠較好地應(yīng)對(duì)變化,降本增效。

    而設(shè)計(jì)原則又分為七個(gè)部分,分別是開閉原則、里式替換原則、依賴倒置原則、接口隔離原則、最小知識(shí)原則、單一職責(zé)原則和合成復(fù)用原則。

    3.1 開閉原則

    第一個(gè)設(shè)計(jì)原則是開閉原則,開閉原則簡(jiǎn)稱OCP,正如英文定義的那樣the open–closed principle,?Entities should be open for extension, but closed for modification??對(duì)擴(kuò)展開放,對(duì)修改關(guān)閉。

    這樣做的目的是保護(hù)已有代碼的穩(wěn)定性、復(fù)用性、靈活性、維護(hù)性和可擴(kuò)展性,同時(shí)又讓系統(tǒng)更具有彈性。

    Android需求發(fā)生變化的時(shí)候,不提倡直接修改Android基類源代碼,盡量擴(kuò)展模塊或者擴(kuò)展原有的功能,去實(shí)現(xiàn)新需求變更。

    關(guān)于開閉原則,一般采用繼承或?qū)崿F(xiàn)的方式,比如: 如果涉及到非通用功能,不要把業(yè)務(wù)邏輯加到BaseActvity,而是單獨(dú)使用ChildActvity類繼承abstract BaseActvity,并讓ChildActvity去拓展abstract BaseActvity的抽象方法。

    翻一翻開源庫(kù)源碼,面向抽象類或面向接口去實(shí)現(xiàn)的功能場(chǎng)景非常常見。

    那么,為什么要使用開閉原則呢?

    第一,開閉原則可以降低功能設(shè)計(jì)成本

    第二,開閉原則可以提高代碼穩(wěn)定性

    第三,開閉原則可以提高代碼可維護(hù)性

    第四,開閉原則可以降低測(cè)試的成本

    因?yàn)闊o(wú)論是大佬還是小白改動(dòng)工程陳舊代碼塊,都無(wú)法保證改完后代碼是0風(fēng)險(xiǎn)的。因此,如果遵守開閉原則,那么可以極大限度的降低變更引發(fā)的歷史功能性缺失、邏輯漏洞等風(fēng)險(xiǎn)。

    3.1.1 UML圖例

    老爸幫小明去買書,書有很多特征,一種特征是書是有名字的,一種特征是書是有價(jià)格的,那如果按照開閉原則的話,首先要定義一個(gè)IBook接口,描述書的兩種特征:名稱、價(jià)格。

    然后用一個(gè)類NovelBook去實(shí)現(xiàn)這個(gè)接口,方便讀取和修改書的名稱和價(jià)格。

    根據(jù)開閉原則,使用者如果要對(duì)書進(jìn)行比如打折降價(jià)活動(dòng)是不能直接在NovelBook操作的,需要用DisNovelBook繼承NovelBook去拓展NovelBook的getName和getPrice方法。

    3.1.2 Bad Code

    //----------------------------代碼片段一----------------------------/*** 功能描述: 定義小說(shuō)類NovelBook-實(shí)現(xiàn)類*/ public class NovelBook implements IBook {public String name;public int price;public NovelBook(String name, int price) {this.name = name;this.price = price;}@Overridepublic String getName() {return this.name;}@Overridepublic int getPrice() {if (this.price > 50) {return (int) (this.price * 0.9);} else {return this.price;}} } //----------------------------代碼片段二---------------------------- /**** 功能描述: 現(xiàn)在有個(gè)書店售書的場(chǎng)景,首先定義一個(gè)IBook類,里面有兩個(gè)屬性:名稱、價(jià)格。*/ public interface IBook{public String getName();public int getPrice(); } //----------------------------代碼片段三---------------------------- public class Client {public static void main(String[] args) {NovelBook novel = new NovelBook("笑傲江湖", 100);System.out.println("書籍名字:" + novel.getName() + "書籍價(jià)格:" + novel.getPrice());} } 復(fù)制代碼

    3.1.3 Good Code

    因?yàn)槿绻磥?lái)需求變更,如小明要買數(shù)學(xué)書和化學(xué)書,其中化學(xué)書價(jià)格不能超過(guò)15元,數(shù)學(xué)不能高于30元,且數(shù)學(xué)書可以使用人教版,而化學(xué)書既可以使用湘教版也可以使用人教版。

    //----------------------------代碼片段一----------------------------/*** 功能描述: 定義小說(shuō)類NovelBook-實(shí)現(xiàn)類*/ public class NovelBook implements IBook {public String name;public int price;public NovelBook(String name, int price) {this.name = name;this.price = price;}@Overridepublic String getName() {return this.name;}@Overridepublic int getPrice() {return this.price;} } //----------------------------代碼片段二----------------------------public class DisNovelBook extends NovelBook {public DisNovelBook(String name, int price) {super(name, price);}// 復(fù)寫價(jià)格方法,當(dāng)價(jià)格大于50,就打9析@Overridepublic int getPrice() {if (this.price > 50) {return (int) (this.price * 0.9);} else {return this.price;}} }//----------------------------代碼片段三---------------------------- /**** 功能描述: 現(xiàn)在有個(gè)書店售書的場(chǎng)景,首先定義一個(gè)IBook類,里面有兩個(gè)屬性:名稱、價(jià)格。*/ public interface IBook{public String getName();public int getPrice(); }//----------------------------代碼片段四---------------------------- public class Client{public static void main(String[] args){IBook disnovel = new DisNovelBook ("小木箱成長(zhǎng)營(yíng)",100000);System.out.println("公眾號(hào)名字:"+disnovel .getName()+"公眾號(hào)粉絲數(shù)量:"+disnovel .getPrice());} } 復(fù)制代碼

    這些邏輯加在一塊的話,因?yàn)橘?gòu)買條件不一樣,需要將不變的邏輯抽象成接口實(shí)現(xiàn)類NovelBook,但如果不使用開辟原則,直接更改接口實(shí)現(xiàn)類NovelBook,隨著需求不斷膨脹,但凡多增加一些控制項(xiàng),在多人協(xié)同開發(fā)過(guò)程中代碼維護(hù)風(fēng)險(xiǎn)度會(huì)越來(lái)越高。

    3.1.4 使用原則

    開辟原則使用原則有2個(gè)點(diǎn),第一個(gè)點(diǎn)是抽象約束;第二個(gè)點(diǎn)是封裝變化

    首先來(lái)說(shuō)一說(shuō)抽象約束,抽象約束一共有三個(gè)方面,第一個(gè)方面是接口或抽象類的方法全部要public,方便去使用。

    第二個(gè)方面是參數(shù)類型、引用對(duì)象盡量使用接口或者抽象類,而不是實(shí)現(xiàn)類;因?yàn)槭褂媒涌诤统橄箢惪梢员苊庹J(rèn)為更改起始數(shù)據(jù);

    第三點(diǎn)是抽象層盡量保持穩(wěn)定,一旦確定即不允許修改。如果抽象層經(jīng)常變更,會(huì)導(dǎo)致所有實(shí)現(xiàn)類報(bào)錯(cuò)。

    接著來(lái)說(shuō)一說(shuō)封裝變化,封裝變化一共有兩個(gè)方面,第一個(gè)方面是相同的邏輯要抽象到一個(gè)接口或抽象類中。

    第二個(gè)方面是將不同的變化封裝到不同的接口或抽象類中,不應(yīng)該有兩個(gè)不同的變化出現(xiàn)在同一個(gè)接口或抽象類中。

    比如上文說(shuō)的,如果老爸買完書了,準(zhǔn)備買菜,那么要單獨(dú)立一個(gè)IVegetable的接口。而不是改造原來(lái)的IBook。

    3.2 里氏替換原則

    第二個(gè)設(shè)計(jì)原則是里氏替換原則,里氏替換原則簡(jiǎn)稱LSP,正如英文定義的那樣The Liskov Substitution Principle,Functions that use pointers of references to base classes must be able to use objects of derived classes without knowing it 。

    子類可以替換父類,子類對(duì)象能夠替換程序中父類對(duì)象出現(xiàn)的任何地方,并且保證原來(lái)程序的邏輯行為不變以及正確性不會(huì)被破壞。

    相當(dāng)于子類可以擴(kuò)展父類功能。繼承是里氏替換原則的重要表現(xiàn)方式。里氏替換原則用來(lái)指導(dǎo)繼承關(guān)系中子類該如何設(shè)計(jì)的。

    里氏替換原則,注意事項(xiàng)是盡量不要重寫父類的方法,也是開閉原則的重要方式之一,為什么不建議重寫父類的方法呢?

    因?yàn)橹貙憰?huì)覆蓋父類的功能,導(dǎo)致使用者對(duì)類預(yù)期功能被修改后得到就不是對(duì)方想要的功能。

    提出問(wèn)題

    下面有個(gè)關(guān)于Bird鳥類位移時(shí)間的技術(shù)需求:

    已知Bird(基類)的子類Swallow(小燕子)和Ostrich(鴕鳥)位移了300米,Ostrich(鴕鳥)的跑步速度為120米/秒。

    Swallow(小燕子)和Ostrich(鴕鳥)的飛行速度為120米/秒和0米/秒,求Swallow(小燕子)和Ostrich(鴕鳥)的位移時(shí)間。

    分析問(wèn)題

    位移時(shí)間,算的是位移距離/跑步速度還是位移距離/飛行速度呢?

    Ostrich(鴕鳥)能飛嗎?

    Swallow(小燕子)飛行速度能否單獨(dú)抽象成一個(gè)方法呢?

    解決問(wèn)題

    可以參考UML圖例、Good Code、Bad Code和里氏替換使用原則。

    3.2.1 UML圖例

    3.2.2 Bad Code

    常規(guī)方式?: 定義鳥的基類Bird,Bird(基類)有一個(gè)setFlySpeed(飛翔速度)。根據(jù)distance(距離)去算出它飛翔的getFlyTime(飛翔時(shí)間)。

    //----------------------------代碼片段一---------------------------- public class Bird {private double flySpeed;public void setFlySpeed(double speed) {this.flySpeed = speed;}public double getFlyTime(double distance) {return distance / flySpeed;} }//----------------------------代碼片段二---------------------------- public class Swallow extends Bird {} //----------------------------代碼片段三---------------------------- public class Ostrich extends Bird{@Overridepublic void setFlySpeed(double speed) {speed = 0;} } //----------------------------代碼片段四---------------------------- public class Main {public static void main(String[] args) {Bird swallow = new Swallow();Bird ostrich = new Ostrich();swallow.setFlySpeed(120);ostrich.setFlySpeed(120);System.out.println("小木箱說(shuō),如果飛行300公里:");try {System.out.println("燕子將飛行: " + swallow.getFlyTime(300) + "小時(shí)。"); // 燕子飛行2.5小時(shí)。System.out.println("鴕鳥將飛行: " + ostrich.getFlyTime(300) + "小時(shí)。"); // 鴕鳥將飛行Infinity小時(shí)。} catch (Exception err) {System.out.println("發(fā)生錯(cuò)誤了!");}} } 復(fù)制代碼

    Bird(基類)有兩個(gè)子類,一個(gè)是Swallow(小燕子),一個(gè)是Ostrich(鴕鳥)。

    小燕子只要設(shè)置正確的setFlySpeed(速度)和distance(距離)即可。

    但Ostrich(鴕鳥)不太一樣,Ostrich(鴕鳥)是不會(huì)飛的,Ostrich(鴕鳥)只會(huì)地上跑。

    因?yàn)镺strich(鴕鳥)沒(méi)有flySpeed(飛翔速度)。那在構(gòu)造Ostrich(鴕鳥),去繼承實(shí)現(xiàn)這 Bird(基類), Ostrich(鴕鳥)的重寫方法setFlySpeed(設(shè)置飛翔速度)傳0.0。

    在Bird(基類) 當(dāng)中去計(jì)算getFlyTime(飛翔時(shí)間),按照常規(guī)的應(yīng)該distance(距離) / setFlySpeed(設(shè)置飛翔速度),就得到了getFlyTime(飛翔時(shí)間)。

    去調(diào)用getFlyTime(飛翔時(shí)間) 時(shí)間的時(shí)候,因?yàn)閷?duì)Ostrich(鴕鳥) 的getFlyTime(飛翔時(shí)間)的子類的參數(shù)speed,重寫了setFlySpeed(設(shè)置飛翔速度)方法,并設(shè)置該方法speed參數(shù)為0,數(shù)學(xué)里面0不能作為分母,所以會(huì)得到一個(gè)無(wú)效結(jié)果Infinity,重寫過(guò)程,違背了里氏替換原則。

    結(jié)果:

    3.2.3 Good Code

    正確的方式是??:打斷Ostrich(鴕鳥)和Bird(基類)繼承關(guān)系,定義Bird(基類)和Ostrich(鴕鳥)的超級(jí)父類Animal(動(dòng)物),讓Animal(動(dòng)物)有奔跑能力。Ostrich(鴕鳥)的飛行速度雖然為 0,但奔跑速度不為 0,可以計(jì)算出其奔跑 300 千米所要花費(fèi)的時(shí)間。

    那么,雖然不能將Ostrich(鴕鳥)的getRunTime(位移時(shí)間)抽象成 Bird(基類)的 getFlyTime(飛翔時(shí)間)。

    但可以利用超級(jí)父類Animal(動(dòng)物)的getRunTime(位移時(shí)間),即花費(fèi)時(shí)長(zhǎng),這時(shí)Ostrich(鴕鳥)的setRunSpeed(跑步速度)就不為0,因?yàn)镺strich(鴕鳥)復(fù)用了超級(jí)父類Animal(動(dòng)物) getRunTime(位移時(shí)間)功能。

    超級(jí)父類Animal(動(dòng)物)有一個(gè) getRunSpeed(跑步速度) ,而不是Bird(基類)的setFlySpeed那個(gè)飛翔速度。

    去設(shè)置setRunSpeed(跑步速度) 之后。因?yàn)槲灰剖莿?dòng)物的天性。鳥類和鴕鳥都具備位移能力。

    所以可以在超級(jí)父類Animal(動(dòng)物) 的基礎(chǔ)上,定義Bird(基類) 子類,去繼承 Animal(動(dòng)物) ,把Animal(動(dòng)物)的一些能力轉(zhuǎn)化成Bird(基類) 相關(guān)一些能力,這樣就和預(yù)期需求是一致的了。

    //----------------------------代碼片段一---------------------------- public class Bird extends Animal {private double flySpeed;public void setFlySpeed(double speed) {this.flySpeed = speed;}public double getFlyTime(double distance) {return distance / flySpeed;} } //----------------------------代碼片段二---------------------------- public class Animal {private double runSpeed;public double getRunTime(double distance) {return distance / speed;}public void setRunSpeed(double speed) {this.runSpeed = speed;}} //----------------------------代碼片段三---------------------------- public class Swallow extends Bird {}//----------------------------代碼片段四---------------------------- public class Ostrich extends Animal{} //----------------------------代碼片段五---------------------------- public class Main {public static void main(String[] args) {Bird swallow = new Swallow();Animal ostrich = new Ostrich();swallow.setFlySpeed(120);ostrich.setRunSpeed(120);System.out.println("如果飛行300公里:");try {System.out.println("燕子將位移: " + swallow.getFlyTime(300) + "小時(shí)。"); System.out.println("鴕鳥將位移: " + ostrich.getRunTime(300) + "小時(shí)。"); } catch (Exception err) {System.out.println("發(fā)生錯(cuò)誤了!");}} } 復(fù)制代碼

    結(jié)果:

    3.2.4 使用原則

    Java中,多態(tài)是不是違背了里氏替換原則?

    那么,JAVA中,多態(tài)是不是違背了里氏替換原則呢?如果extends的目的是為了多態(tài),而多態(tài)的前提就是Swallow(子類)覆蓋并重新定義Bird(基類)的getFlySpeed()。

    為了符合LSP,應(yīng)該將Bird(基類)定義為abstract,并定義getFlySpeed()(抽象方法),讓Swallow(子類)重新定義getFlySpeed()。

    當(dāng)父類是abstract時(shí),Bird(基類)就是不能實(shí)例化,所以也不存在可實(shí)例化的Bird(基類)對(duì)象在程序里。

    //----------------------------代碼片段一---------------------------- public abstract class Bird{protected abstract double getFlySpeed();public double getFlyTime(double distance){return distance / getFlySpeed();}}//----------------------------代碼片段二----------------------------public class Swallow extends Bird{protected double getFlySpeed(){return 100.0;}} 復(fù)制代碼

    里氏替換原則和開閉原則的區(qū)別有哪些?

    里氏替換原則和開閉原則的區(qū)別在于: 開閉原則大部分是面向接口編程,少部分是針對(duì)繼承的,而里氏替換原則主要針對(duì)繼承的,降低繼承帶來(lái)的復(fù)雜度

    什么時(shí)候使用里氏替換原則?

    使用里氏替換原則的時(shí)機(jī)有兩個(gè),第一個(gè)是重新提取公共部分的方法,第二個(gè)是改變繼承關(guān)系.

    首先,重新提取公共部分的方法主要是把公共部分提取出來(lái)作為一個(gè)抽象基類.

    而提取公共部分的時(shí)機(jī)是代碼不是很多的時(shí)候應(yīng)用,提取得部分可以作為一個(gè)設(shè)計(jì)工具.

    然后,改變繼承關(guān)系主要是從父子關(guān)系變?yōu)槲申P(guān)系或兄弟關(guān)系,可以把它們的一些公有特性提取到一個(gè)抽象接口,再分別實(shí)現(xiàn).具體可以看 #3.2.1 UML圖例

    3.3 依賴倒置原則

    第三個(gè)設(shè)計(jì)原則是里氏替換原則,里氏替換原則簡(jiǎn)稱DIP,正如英文定義的那樣Dependence Inversion Principle,Abstractions should not depend on details. Details should depend on abstractions,抽象不依賴于細(xì)節(jié),而細(xì)節(jié)依賴于抽象。高層模塊不能直接依賴低層模塊,而是通過(guò)接口或抽象的方式去實(shí)現(xiàn)。

    從定義也就可以看出來(lái),依賴倒置原則是為了降低類或模塊的耦合性,提倡面向接口編程,能降低工程維護(hù)成本,降低由于類或?qū)崿F(xiàn)發(fā)生變化帶來(lái)的修改成本,提高代碼穩(wěn)定性。

    比如小木箱在組件化設(shè)計(jì)當(dāng)中,會(huì)員模塊、訂單模塊和用戶模塊不應(yīng)該直接依賴基礎(chǔ)平臺(tái)組件數(shù)據(jù)庫(kù)、網(wǎng)絡(luò)和統(tǒng)計(jì)組件等。

    而應(yīng)該從會(huì)員模塊、訂單模塊和用戶模塊抽取BaseModule和中間件等模塊,橫向依賴基礎(chǔ)平臺(tái)組件BaseModule和中間件,去實(shí)現(xiàn)模塊與模塊之間的一些訪問(wèn)與跳轉(zhuǎn),這樣層級(jí)才會(huì)更清晰。

    依賴倒置原則核心思想是面向接口編程,因?yàn)槿绻嫦驅(qū)崿F(xiàn)類,實(shí)現(xiàn)類如果發(fā)生變化,那么依賴實(shí)現(xiàn)類的實(shí)現(xiàn)方法和功能都會(huì)產(chǎn)生蝴蝶效應(yīng)。

    提出問(wèn)題

    小木箱剛拿到駕照,準(zhǔn)備在電動(dòng)車、新能源、汽油車三類型進(jìn)行購(gòu)車,于是拿沃爾沃、寶馬、特斯拉進(jìn)行測(cè)試,請(qǐng)用代碼讓這三輛汽車自動(dòng)跑起來(lái)?

    分析問(wèn)題

    如果小木箱想把跑起來(lái)的自動(dòng)駕駛代碼,復(fù)用給其他駕駛者,代碼的健壯性如何?

    解決問(wèn)題

    可以參考UML圖例、Good Code、Bad Code和思考復(fù)盤。

    3.3.1 UML圖例

    3.3.2 Bad Code

    下面代碼比較劣質(zhì)的原因在于自動(dòng)駕駛能力與駕駛者高耦合度,如果想讓其他駕駛者使用自動(dòng)駕駛系列的車,那么駕駛者必須將車型實(shí)例重新傳給其他駕駛者,沒(méi)有做到真正意義上的插拔式注冊(cè),換個(gè)駕駛者就不成立了。

    //----------------------------代碼片段一---------------------------- public class BMW {public void autoRun() {System.out.println("BMW is running!");} } //----------------------------代碼片段二---------------------------- public class Tesla {public void autoRun() {System.out.println("Tesla is running!");} } //----------------------------代碼片段三---------------------------- public class Volvo {public void autoRun() {System.out.println("Volvo is running!");} } //----------------------------代碼片段四---------------------------- public class AutoDriver {public void autoDrive(Tesla tesla) {tesla.autoRun();}public void autoDrive(BMW bm) {bm.autoRun();}public void autoDrive(Volvo volvo) {volvo.autoRun();}} //----------------------------代碼片段四---------------------------- public class Main {public static void main(String[] args) {Tesla tesla = new Tesla();BMW bm = new BMW();Volvo volvo = new Volvo();AutoDriver driver = new AutoDriver();driver.autoDrive(tesla);driver.autoDrive(bm);driver.autoDrive(volvo);} } 復(fù)制代碼

    結(jié)果:

    3.3.3 Good Code

    那么,正確實(shí)現(xiàn)方式是怎樣的呢? 首先要定義一個(gè)自動(dòng)駕駛接口IAutoDriver。因?yàn)樽詣?dòng)駕駛,新能源比如說(shuō)像寶馬、特斯拉、沃爾沃都有實(shí)現(xiàn)自動(dòng)駕駛能力。

    但是比如說(shuō)像紅旗、長(zhǎng)城不是一個(gè)自動(dòng)駕駛的實(shí)現(xiàn)者。

    那對(duì)自動(dòng)駕駛接口IAutoDriver,如果你有自動(dòng)駕駛能力,那么你就去實(shí)現(xiàn)IAutoDriver,去重寫autoDrive(自動(dòng)駕駛)的能力。否則,就不實(shí)現(xiàn)自動(dòng)駕駛IAutoDriver接口。

    對(duì) AutoDriver 的話,駕駛者是去通過(guò)依賴倒置原則,把寶馬、特斯拉、沃爾沃自動(dòng)駕駛模式接口IAutoCar傳進(jìn)來(lái),通過(guò)autoRun開啟自動(dòng)駕駛模式。

    autoRun是區(qū)分了自動(dòng)駕駛還是普通駕駛模式。具體的代碼方式很簡(jiǎn)單,首先 new一個(gè)寶馬實(shí)例,然后去實(shí)現(xiàn)自動(dòng)駕駛接口 IAutoCar,最后把寶馬實(shí)例傳給 AutoDriver,實(shí)現(xiàn)自動(dòng)駕駛的方式,特斯拉、沃爾沃也是這樣的。

    對(duì)于自動(dòng)駕駛技術(shù),不關(guān)心駕駛的什么車,寶馬、特斯拉、沃爾沃還是大眾,只關(guān)心你是實(shí)現(xiàn)了IAutoDriver接口。只關(guān)心你是否有autoDrive(自動(dòng)駕駛)能力。

    如果有自動(dòng)駕駛能力,使用者就直接調(diào)用autoDrive(自動(dòng)駕駛)能力。具體的怎么實(shí)現(xiàn)呢?是AutoDriver的實(shí)現(xiàn)類IAutoDriver決定的,這便是依賴倒置原則,不依賴具體的實(shí)現(xiàn),只調(diào)IAutoCar接口方法選擇自動(dòng)駕駛模式autoRun即可.

    //----------------------------代碼片段一---------------------------- public interface IAutoCar {public void autoRun(); } //----------------------------代碼片段二---------------------------- public class BMW implements IAutoCar{@Overridepublic void autoRun() {System.out.println("BMW is running!");} } //----------------------------代碼片段三---------------------------- public class Tesla implements IAutoCar {@Overridepublic void autoRun() {System.out.println("Tesla is running!");} } //----------------------------代碼片段四---------------------------- public class Volvo implements IAutoCar{@Overridepublic void autoRun() {System.out.println("Volvo is running!");} }//----------------------------代碼片段五---------------------------- public interface IAutoDriver {public void autoDrive(IAutoCar car); } //----------------------------代碼片段六---------------------------- public class AutoDriver implements IAutoDriver{@Overridepublic void autoDrive(IAutoCar car) {car.autoRun();} } //----------------------------代碼片段六---------------------------- public class Main {public static void main(String[] args) {IAutoDriver driver = new AutoDriver();driver.autoDrive(new Tesla());driver.autoDrive(new BMW());driver.autoDrive(new Volvo());} } 復(fù)制代碼

    結(jié)果:

    3.3.4 使用原則

    在簡(jiǎn)單工廠設(shè)計(jì)模式和策略設(shè)計(jì)模式,都是使用依賴倒置原則進(jìn)行注入,不過(guò)簡(jiǎn)單工廠設(shè)計(jì)模式, 使用的是接口方法注入, 而策略設(shè)計(jì)模式使用的是構(gòu)造函數(shù)注入,這一塊后文詳細(xì)介紹。

    3.4 單一職責(zé)原則

    第四個(gè)設(shè)計(jì)原則是單一職責(zé)原則,單一職責(zé)原則簡(jiǎn)稱SRP, 正如英文The Single Responsibility Principle定義的那樣,A class should have one, and only one, reason to change。

    單一職責(zé)指的是一個(gè)類只能因?yàn)橐粋€(gè)理由被修改,一個(gè)類只做一件事。不要設(shè)計(jì)大而全的類,要設(shè)計(jì)粒度小、功能單一的類。

    類的職能要有界限。單一原則要求類要高內(nèi)聚,低耦合。意思是為了規(guī)避代碼冗余,無(wú)關(guān)職責(zé)、無(wú)關(guān)功能的方法和對(duì)象不要引入類里面。

    因?yàn)槿绻粋€(gè)類承擔(dān)的職責(zé)過(guò)多,就等于把這些職責(zé)耦合在一起,一個(gè)職責(zé)的變化可能會(huì)削弱或者抑制這個(gè)類完成其他職責(zé)的能力。

    這種耦合會(huì)導(dǎo)致脆弱他的設(shè)計(jì),當(dāng)變化發(fā)生時(shí),設(shè)計(jì)會(huì)遭受到意想不到的破壞;軟件設(shè)計(jì)真正要做的許多內(nèi)容就是發(fā)現(xiàn)職責(zé)并把那些職責(zé)相互分離。

    比如去銀行取錢,取錢的類不應(yīng)該包含打印發(fā)票,取錢的類只管取錢動(dòng)作,打印發(fā)票功能,需要新建類完成。目的是降低類的復(fù)雜度,提高閱讀性,降低代碼變更造成的風(fēng)險(xiǎn)。

    再比如Android里面Activity過(guò)于臃腫會(huì)讓感覺很頭大,MVP、MVVM、MVP和MVI等架構(gòu)都是為了讓Activity變得職責(zé)單一。

    提出問(wèn)題:

    老師去網(wǎng)上采購(gòu)“ 三國(guó)演義 ”、“ 紅樓夢(mèng) ”、“ 三國(guó)演義 ”、“ 西游記 ”各一本。

    已知“ 紅樓夢(mèng) ”50元/本,“ 三國(guó)演義 ”40元/本,“ 西游記 ”30元/本,“ 水滸傳 ”20元/本。

    如果“ 紅樓夢(mèng) ” 8 折促銷,“ 西游記 ”6 折促銷,根據(jù)書的價(jià)格,求所有圖書的總價(jià)格。

    分析問(wèn)題:

    如果采購(gòu)1000本書籍,單品折扣策閱可能不一樣,如果單品價(jià)格隨著單品購(gòu)買數(shù)量變化,那么購(gòu)物車價(jià)格條件一旦變化,購(gòu)物車代碼會(huì)因此膨脹,進(jìn)而影響代碼可維護(hù)性,如何解決這種問(wèn)題?

    3.4.1 UML圖例

    3.4.2 Bad Code

    這段壞味道的代碼問(wèn)題就在于: 購(gòu)物車摻雜了價(jià)格計(jì)算功能,購(gòu)物車正常只關(guān)心對(duì)商品的CRUD能力,如果有一天,價(jià)格計(jì)算方式改變,那這里就需要?jiǎng)淤?gòu)物車代碼,購(gòu)物車變更會(huì)引起方法變動(dòng),從而帶來(lái)風(fēng)險(xiǎn)。

    //----------------------------代碼片段一---------------------------- public class WoodBook {private String name;private double price;public WoodBook(String name, double price) {this.name = name;this.price = price;}public String getName() {return name;}public double getPrice() {return price;} } //----------------------------代碼片段二----------------------------public class ShoppingCart { private List<WoodBook> list = new ArrayList<>(); public void addBook(WoodBook book) {list.add(book);}public double checkOut() {double total = 0;for (WoodBook book : list) {if ("紅樓夢(mèng)".equals(book.getName())) {total = total + book.getPrice() * 0.8;} else if ("西游記".equals(book.getName())) {total = total + book.getPrice() * 0.6;} else {total = total + book.getPrice();}}return total;} } //----------------------------代碼片段三---------------------------- public class Main {public static void main(String[] args) {ShoppingCart shoppingCart = new ShoppingCart();shoppingCart.addBook(new WoodBook("紅樓夢(mèng)",50));shoppingCart.addBook(new WoodBook("三國(guó)演義",40));shoppingCart.addBook(new WoodBook("西游記",30));shoppingCart.addBook(new WoodBook("水滸傳",20));double total = shoppingCart.checkOut();System.out.println("所有圖書價(jià)格為:"+total);} } 復(fù)制代碼

    3.4.3 Good Code

    正確的方式: 首先計(jì)算價(jià)格的邏輯,交給接口實(shí)現(xiàn),購(gòu)物車只關(guān)心價(jià)格計(jì)算的結(jié)果,并將結(jié)果返回即可。然后計(jì)算價(jià)格接口交給調(diào)用方實(shí)現(xiàn),使用者不關(guān)心紅樓夢(mèng)和西游記價(jià)格折扣策閱還是0折扣策閱,最后需求如果發(fā)生變更,那么只需要更改調(diào)用方實(shí)現(xiàn)邏輯即可。

    //----------------------------代碼片段一----------------------------public class WoodBook {private String name;private double price;public WoodBook(String name, double price) {this.name = name;this.price = price;}public String getName() {return name;}public double getPrice() {return price;} } //----------------------------代碼片段二---------------------------- public class DefaultDiscountStrategy implements DiscountStrategy {@Overridepublic double discount(List<WoodBook> list) {double total = 0;for (WoodBook book : list) {total = total + book.getPrice();}return total;} } //----------------------------代碼片段三---------------------------- public class SingleDiscountStrategy implements DiscountStrategy {@Overridepublic double discount(List<WoodBook> list) {double total = 0;for (WoodBook book : list) {if ("西游記".equals(book.getName())) {total = total + book.getPrice() * 0.6;} else if ("紅樓夢(mèng)".equals(book.getName().toString())) {total = total + book.getPrice() * 0.8;}else {total = total + book.getPrice() ;}}return total;} } //----------------------------代碼片段四---------------------------- public class ShoppingCart {private List<WoodBook> list = new ArrayList<>();private DiscountStrategy discountStrategy;public void addBook(WoodBook book) {list.add(book);}public void setDiscountStrategy(DiscountStrategy discountStrategy) {this.discountStrategy = discountStrategy;}public double checkOut() {if (discountStrategy == null) {discountStrategy = new DefaultDiscountStrategy();}return discountStrategy.discount(list);} } //----------------------------代碼片段五---------------------------- public interface DiscountStrategy {double discount(List<WoodBook> list); } //----------------------------代碼片段六---------------------------- public class Main {public static void main(String[] args) {ShoppingCart shoppingCart = new ShoppingCart();shoppingCart.addBook(new WoodBook("紅樓夢(mèng)",50));shoppingCart.addBook(new WoodBook("三國(guó)演義",40));shoppingCart.addBook(new WoodBook("西游記",30));shoppingCart.addBook(new WoodBook("水滸傳",20));shoppingCart.setDiscountStrategy(new SingleDiscountStrategy());double total = shoppingCart.checkOut();System.out.println("所有圖書價(jià)格為:"+total);} } 復(fù)制代碼

    結(jié)果:

    3.4.4 思考復(fù)盤

    關(guān)于單一職責(zé)原則我們有四個(gè)問(wèn)題需要思考

    問(wèn)題一: 如何判斷類的職責(zé)是否足夠單一?

    判斷類的職責(zé)是否足夠單一有五條規(guī)則:

    規(guī)則一: 如果類中的代碼行數(shù)、函數(shù)或?qū)傩赃^(guò)多,會(huì)影響代碼的可讀性和可維護(hù)性,那么我們就需要考慮對(duì)類進(jìn)行拆分;

    規(guī)則二: 如果類依賴的其他類過(guò)多,或者依賴類的其他類過(guò)多,不符合高內(nèi)聚、低耦合的設(shè)計(jì)思想,那么我們就需要考慮對(duì)類進(jìn)行拆分;

    規(guī)則三: 如果私有方法過(guò)多,我們就要考慮能否將私有方法獨(dú)立到新的類中,那么我們就設(shè)置為 public 方法,供更多的類使用,從而提高代碼的復(fù)用性;

    規(guī)則四: 如果比較難給類起一個(gè)合適名字,很難用一個(gè)業(yè)務(wù)名詞概括,或者只能用一些籠統(tǒng)的Manager、Context 之類的詞語(yǔ)來(lái)命名,那么這就說(shuō)明類的職責(zé)定義得可能不夠清晰

    規(guī)則五: 如果類中大量的方法都是集中操作類中的某幾個(gè)屬性,比如: 在 UserInfo 例子中,如果一半的方法都是在操作 address 信息,那么可以考慮將這幾個(gè)屬性和對(duì)應(yīng)的方法拆分出來(lái)

    問(wèn)題二: 類的職責(zé)是否設(shè)計(jì)得越單一越好?

    類的職責(zé)單一性標(biāo)準(zhǔn)有四方面。

    第一方面,單一職責(zé)原則通過(guò)避免設(shè)計(jì)大而全的類,避免將不相關(guān)的功能耦合在一起,來(lái)提高類的內(nèi)聚性。

    第二方面,類職責(zé)單一,類依賴的和被依賴的其他類也會(huì)變少,減少了代碼的耦合性,以此來(lái)實(shí)現(xiàn)代碼的高內(nèi)聚、低耦合。

    第三方面,如果拆分得過(guò)細(xì),實(shí)際上會(huì)適得其反,反倒會(huì)降低內(nèi)聚性,也會(huì)影響代碼的可維護(hù)性。

    第四方面,根據(jù)不同的場(chǎng)景對(duì)某個(gè)類或模塊單一職責(zé)的判斷是不同的,不能為了拆分而拆分,造成過(guò)度設(shè)計(jì),難以維護(hù)。

    問(wèn)題三: 單一職責(zé)原則為什么要這么設(shè)計(jì)?

    那么單一職責(zé)原則為什么要這么設(shè)計(jì)?因?yàn)槿绻粋€(gè)類承擔(dān)的職責(zé)過(guò)多,即耦合性太高一個(gè)職責(zé)的變化可能會(huì)影響到其他的職責(zé)。

    問(wèn)題四: Hook違背了單一職責(zé)原則嗎?

    那么,Hook違背了單一職責(zé)原則嗎?Hook突破了Java層OOP系統(tǒng)層設(shè)計(jì)理念,也就違背了單一職責(zé)原則。Hook雖好,不建議廣泛使用,因?yàn)樵陂_發(fā)過(guò)程中可能導(dǎo)致依賴不清晰、命名沖突、來(lái)源不清晰等問(wèn)題。

    3.5 接口隔離原則

    第五個(gè)原則是接口隔離原則,接口隔離原則指的是接口隔離原則是指客戶端不應(yīng)該依賴于它不需要的接口。接口隔離原則簡(jiǎn)稱ISP,正如英文定義的那樣interface-segregation principle,Clients should not be forced to depend upon interfaces that they do not use. 客戶端不應(yīng)該被強(qiáng)迫依賴它不需要的接口。其中的 “客戶端”,可以理解為接口的調(diào)用者或者使用者。

    接口隔離原則是盡量將臃腫龐大的接口顆粒度拆得更細(xì)。和單一原則類似,一個(gè)接口,涵蓋的職責(zé)實(shí)現(xiàn)的功能盡量簡(jiǎn)單單一,只跟接口自身想實(shí)現(xiàn)的功能相關(guān),不能把別人干的活也涵蓋進(jìn)來(lái),讓實(shí)現(xiàn)者只關(guān)心接口獨(dú)立單元方法。

    我在架構(gòu)組設(shè)計(jì)對(duì)外的 API 或?qū)ν饽芰?#xff0c;接口干的職責(zé),要非常明確的,接口不能做與接口無(wú)關(guān)工作或隱藏邏輯,一個(gè)類對(duì)一個(gè)類依賴應(yīng)建立在最小接口依賴基礎(chǔ)之上。

    提出問(wèn)題

    小木箱是一名AndroidDev也是一名DevopsDev,請(qǐng)用代碼分類打印標(biāo)記小木箱的技能樹。

    分析問(wèn)題

    首先將技能樹全部存放到技能清單IDev,然后讓AndroidDev和DevopsDev分別實(shí)現(xiàn)技能清單IDev,最后在AndroidDev和DevopsDev匹配的技能樹打印標(biāo)記。

    解決問(wèn)題

    可以參考UML圖例、Good Code、Bad Code和接口隔離使用原則。

    3.5.1 UML圖例

    3.5.2 Bad Code

    比如小木箱做了AndroidDev和DevopsDev兩份簡(jiǎn)歷,而AndroidDev簡(jiǎn)歷和DevopsDev簡(jiǎn)歷所具備的技術(shù)棧又各不相同,但歸檔在小木箱同一份IDev技能樹清單里面。

    如果小木箱把AndroidDev簡(jiǎn)歷和DevopsDev簡(jiǎn)歷實(shí)現(xiàn)技能樹清單接口,那么勢(shì)必會(huì)導(dǎo)致AndroidDev簡(jiǎn)歷既有Devops簡(jiǎn)歷也有AndroidDev技能樹,DevopsDev簡(jiǎn)歷既有DevopsDev技能樹也有AndroidDev技能樹。

    如果有一天小木箱技能樹清單接口技能發(fā)生相應(yīng)的變化,那么很容易給兩份簡(jiǎn)歷帶來(lái)一些風(fēng)險(xiǎn)和改變。

    //--------------------------------代碼塊一--------------------------------- public interface IDev {void framework();void ci2cd();void jetpack();void java(); } //--------------------------------------代碼塊二--------------------------------------- public class AndroidDev implements IDev{@Overridepublic void framework() {System.out.println("CrazyCodingBoy is a Android developer and he knows framework");}@Overridepublic void ci2cd() {}@Overridepublic void jetpack() {System.out.println("CrazyCodingBoy is a Android developer and he knows jetpack");}@Overridepublic void java() {System.out.println("CrazyCodingBoy is a Android developer and he knows java");} } //--------------------------------------代碼塊三--------------------------------------- public class DevopsDev implements IDev {@Overridepublic void framework() {}@Overridepublic void ci2cd() {System.out.println("CrazyCodingBoy is a Devops developer and he knows CI and CD");}@Overridepublic void jetpack() {}@Overridepublic void java() {System.out.println("CrazyCodingBoy is a Devops developer and he knows java");} } //--------------------------------------代碼塊四--------------------------------------- public class Main {public static void main(String[] args) {AndroidDev androidDev = new AndroidDev();DevopsDev devopsDev = new DevopsDev();androidDev.framework();androidDev.jetpack();devopsDev.ci2cd();androidDev.java();devopsDev.java();// TODO: delete 無(wú)效空實(shí)現(xiàn) androidDev.ci2cd(); devopsDev.framework();devopsDev.jetpack();} 復(fù)制代碼

    結(jié)果:

    3.5.3 Good Code

    接口隔離原則是把臃腫龐大的IDev技能樹清單接口,拆分成力度更小的ICi2cd、IFramework、IJetpack和IJava接口,提高整個(gè)系統(tǒng)和接口的一個(gè)靈活性和可維護(hù)性,同時(shí)提高整個(gè)系統(tǒng)內(nèi)聚性,減少對(duì)外交互。

    ICi2cd只關(guān)心小木箱CI/CD的研發(fā)能力,誰(shuí)想持有這個(gè)能力就交給誰(shuí)去實(shí)現(xiàn),不同的技能樹,交給不同的去完成自己的能力。

    否則,IDev接口功能發(fā)生變化,就得去改AndroidDev和DevopsDev的邏輯。

    如果代碼臃腫,代碼量大,那么容易手抖或改了不該改的,造成線上事故。

    如果通過(guò)接口或模塊隔離方式實(shí)現(xiàn),那么就可以降低修改成本。

    //--------------------------------代碼塊一--------------------------------- public interface ICi2cd {void ci2cd(); } //--------------------------------------代碼塊二--------------------------------------- public interface IFramework {void framework(); } //--------------------------------------代碼塊三--------------------------------------- public interface IJetpack {void jetpack(); } //--------------------------------------代碼塊四--------------------------------------- public interface IJava {void java(); } //--------------------------------------代碼塊五--------------------------------------- public class AndroidDev implements IFramework , IJetpack , IJava {@Overridepublic void framework() {System.out.println("CrazyCodingBoy is a Android developer and he knows framework");}@Overridepublic void jetpack() {System.out.println("CrazyCodingBoy is a Android developer and he knows jetpack");}@Overridepublic void java() {System.out.println("CrazyCodingBoy is a Android developer and he knows java");} }//--------------------------------------代碼塊六--------------------------------------- public class DevopsDev implements ICi2cd , IJava {@Overridepublic void ci2cd() {System.out.println("CrazyCodingBoy is a Devops developer and he knows CI and CD");}@Overridepublic void java() {System.out.println("CrazyCodingBoy is a Devops developer and he knows java");} } //--------------------------------------代碼塊七--------------------------------------- public class Main { public static void main(String[] args) {AndroidDev androidDev = new AndroidDev();DevopsDev devopsDev = new DevopsDev();androidDev.framework();androidDev.jetpack();androidDev.java();devopsDev.ci2cd();devopsDev.java();} } 復(fù)制代碼

    結(jié)果:

    3.5.4 思考復(fù)盤

    接著我們聊聊思考復(fù)盤,思考復(fù)盤分為兩方面,第一方面是接口隔離原則和單一職責(zé)原則的區(qū)別?第二方面接口隔離原則優(yōu)點(diǎn)。

    接口隔離原則和單一職責(zé)原則的區(qū)別?

    接口隔離原則和單一職責(zé)原則的區(qū)別有兩個(gè),第一,單一職責(zé)原則指的是類、接口和方法的職責(zé)是單一的,強(qiáng)調(diào)的是職責(zé),也就是說(shuō)在一個(gè)接口里,只要職責(zé)是單一的,有10個(gè)方法也是可以的。

    第二,接口隔離原則指的是在接口中的方法盡量越來(lái)越少,接口隔離原則的前提必須先符合單一職責(zé),在單一職責(zé)的前提下,接口盡量是單一接口。

    接口隔離原則優(yōu)點(diǎn)

    接口隔離原則優(yōu)點(diǎn)有三個(gè)。

    第一,隱藏實(shí)現(xiàn)細(xì)節(jié)

    第二,降低耦合性

    第三,提高代碼的可讀性

    3.6 最小知識(shí)原則

    第六個(gè)設(shè)計(jì)原則是最小知識(shí)原則,最小知識(shí)原則簡(jiǎn)稱LOD,正如英文定義的那樣Law of Demeter

    ,a module should not have knowledge of the inner details of the objects it manipulates?。不該有直接依賴關(guān)系的類,不要有依賴;

    有依賴關(guān)系的類之間,盡量只依賴必要的接口。最小知識(shí)原則是希望減少類之間的耦合,讓類越獨(dú)立越好,每個(gè)類都應(yīng)該少了解系統(tǒng)的其他部分,一旦發(fā)生變化,需要了解這一變化的類就會(huì)比較少。

    最小知識(shí)原則和單一職責(zé)的目的都是實(shí)現(xiàn)高內(nèi)聚低耦合,但是出發(fā)的角度不一樣,單一職責(zé)是從自身提供的功能出發(fā),最小知識(shí)原則是從關(guān)系出發(fā)。

    提出問(wèn)題

    如果我們把一個(gè)對(duì)象看作是一個(gè)人,那么要實(shí)現(xiàn)“一個(gè)人應(yīng)該對(duì)其他人有最少的了解”,做到兩點(diǎn)就足夠了: 第一點(diǎn),只和直接的朋友交流; 第二點(diǎn),減少對(duì)朋友的了解。下面就詳細(xì)說(shuō)說(shuō)如何做到這兩點(diǎn)。

    最小知識(shí)原則還有一個(gè)英文解釋是:talk only to your immediate friends(只和直接的朋友交流)。

    分析問(wèn)題

    什么是朋友呢?每個(gè)對(duì)象都必然會(huì)與其他的對(duì)象有耦合關(guān)系,兩個(gè)對(duì)象之間的耦合就會(huì)成為朋友關(guān)系。

    那么什么又是直接的朋友呢?出現(xiàn)在成員變量、方法的輸入輸出參數(shù)中的類就是直接的朋友。最小知識(shí)原則要求只和直接的朋友通信。

    解決問(wèn)題

    可以參考UML圖例、Good Code、Bad Code和最小知識(shí)原則使用原則。

    3.6.1 UML圖例

    3.6.2 Bad Code

    很簡(jiǎn)單的例子:老師讓班長(zhǎng)清點(diǎn)全班同學(xué)的人數(shù)。這個(gè)例子中總共有三個(gè)類:老師Teacher、班長(zhǎng)GroupLeader和學(xué)生Student。

    在這個(gè)例子中,我們的Teacher有幾個(gè)朋友?兩個(gè),一個(gè)是GroupLeader,它是Teacher的command()方法的入?yún)?#xff1b;另一個(gè)是Student,因?yàn)樵赥eacher的command()方法體中使用了Student。

    那么Teacher有幾個(gè)是直接的朋友?按照直接的朋友的定義

    出現(xiàn)在成員變量、方法的輸入輸出參數(shù)中的類就是直接的朋友

    只有GroupLeader是Teacher的直接的朋友。

    Teacher在command()方法中創(chuàng)建了Student的數(shù)組,和非直接的朋友Student發(fā)生了交流,所以,上述例子違反了最小知識(shí)原則

    方法是類的一個(gè)行為,類竟然不知道自己的行為與其他的類產(chǎn)生了依賴關(guān)系,這是不允許的,嚴(yán)重違反了最小知識(shí)原則

    //--------------------------------------代碼塊一--------------------------------------- public interface IStudent { } //--------------------------------------代碼塊二--------------------------------------- public class Student implements IStudent {} //--------------------------------------代碼塊三--------------------------------------- public interface IGroupLeader {void count(List<Student> students); }//--------------------------------------代碼塊四--------------------------------------- public interface IGroupLeader {void count(List<Student> students); } //--------------------------------------代碼塊五--------------------------------------- public class GroupLeader implements IGroupLeader{@Overridepublic void count(List<Student> students) {System.out.println("The number of students attending the class is: " + students.size());} } //--------------------------------------代碼塊六--------------------------------------- public interface ITeacher {void command(IGroupLeader groupLeader); } //--------------------------------------代碼塊七--------------------------------------- public class Teacher implements ITeacher{@Overridepublic void command(IGroupLeader groupLeader) {List<Student> allStudent = new ArrayList<>();allStudent.add(new Student());allStudent.add(new Student());allStudent.add(new Student());allStudent.add(new Student());allStudent.add(new Student());groupLeader.count(allStudent);} } //--------------------------------------代碼塊八--------------------------------------- public class Main {public static void main(String[] args) {ITeacher teacher = new Teacher();teacher.command(new GroupLeader());} } 復(fù)制代碼

    結(jié)果:

    3.6.3 Good Code

    我們打斷學(xué)生和GroupLeader聯(lián)系,直接的聯(lián)系每個(gè)類都只和直接的朋友交流,有效減少了類之間的耦合

    //--------------------------------------代碼塊一--------------------------------------- public interface IStudent { } //--------------------------------------代碼塊二--------------------------------------- public class Student implements IStudent {} //--------------------------------------代碼塊三--------------------------------------- public interface IGroupLeader {void count(); } //--------------------------------------代碼塊四--------------------------------------- public class GroupLeader implements IGroupLeader {private List<Student> students;public GroupLeader(List<Student> students) {this.students = students;}@Overridepublic void count() {System.out.println("The number of students attending the class is: " + students.size());} } //--------------------------------------代碼塊五--------------------------------------- public interface ITeacher {void command(IGroupLeader groupLeader); } //--------------------------------------代碼塊六--------------------------------------- public class Teacher implements ITeacher {@Overridepublic void command(IGroupLeader groupLeader) {groupLeader.count();} } //--------------------------------------代碼塊七--------------------------------------- public class Main {public static void main(String[] args) {ITeacher teacher = new Teacher();List<Student> allStudent = new ArrayList(4);allStudent.add(new Student());allStudent.add(new Student());allStudent.add(new Student());allStudent.add(new Student());teacher.command(new GroupLeader(allStudent));} } 復(fù)制代碼

    結(jié)果:

    3.6.4 使用原則

    最少知識(shí)原則的使用原則有6個(gè)。

    第一,在類的劃分上,應(yīng)當(dāng)創(chuàng)建弱耦合的類,類與類之間的耦合越弱,就越有利于實(shí)現(xiàn)可復(fù)用的目標(biāo)。 第二,在類的結(jié)構(gòu)設(shè)計(jì)上,每個(gè)類都應(yīng)該降低成員的訪問(wèn)權(quán)限。 第三,在類的設(shè)計(jì)上,只要有可能,一個(gè)類應(yīng)當(dāng)設(shè)計(jì)成不變的類。 第四,在對(duì)其他類的引用上,一個(gè)對(duì)象對(duì)其他類的對(duì)象的引用應(yīng)該降到最低。 第五,盡量限制局部變量的有效范圍,降低類的訪問(wèn)權(quán)限。

    第六,謹(jǐn)慎使用Serializable。

    3.7 合成復(fù)用原則

    最后一個(gè)原則是合成復(fù)用原則。合成復(fù)用原則簡(jiǎn)稱CARP,正如英文定義的那樣Composite/Aggregate Reuse Principle,?try to use composite/aggregate *?*合成復(fù)用原則要求我們?cè)谲浖O(shè)計(jì)的過(guò)程中,盡量不要通過(guò)繼承方式實(shí)現(xiàn)功能和類的一些組合。

    因?yàn)樵?Java 只支持單繼承的, C 、 C ++支持多繼承。所以設(shè)計(jì)模式在 Java 這一塊的規(guī)范,它是不提倡繼承來(lái)解決問(wèn)題的,所以更提倡是合成復(fù)用,一個(gè)類持有另外一個(gè)對(duì)象,把能力交給另外的對(duì)象去完成。

    因?yàn)槔^承破壞了會(huì)繼承復(fù)用的和破壞類的一個(gè)封裝性,子類和父類耦合度會(huì)比較大,因此推薦使用合成復(fù)用原則

    最小知識(shí)原則,如果因?yàn)槭侄?#xff0c;可能會(huì)不小心改了父類,最小知識(shí)原則限制復(fù)用靈活性,合成復(fù)用原則可以維持類的封裝性,降低類與類的耦合度,提高功能的靈活性。

    合成復(fù)用原則可以將已知的對(duì)象和成員變量納入新的對(duì)象和成員變量,方法里邊去調(diào)用成員變量的具體的功能。就達(dá)成了一個(gè)合成復(fù)用原則。

    3.7.1 UML圖例

    3.7.2 Bad Code

    汽車從能源的角度來(lái)說(shuō),分為電動(dòng)車ETCar和汽油車PCar。

    電動(dòng)車ETCar和汽油車PCar有很多顏色,如: 白色、紅色。

    如果后期新增黃色,那么需要電動(dòng)車ETCar和汽油車PCar去繼承Car,并讓紅色車RedPCar和白色車WhiteETCar去繼承電動(dòng)車ETCar和汽油車PCar。繼承的方式可以實(shí)現(xiàn)類組合,但缺點(diǎn)是顏色和車型組合越多,類組合會(huì)呈N 倍遞,導(dǎo)致類爆炸。

    //--------------------------------------代碼塊一--------------------------------------- public abstract class Car {public abstract void move(); } //--------------------------------------代碼塊二--------------------------------------- public abstract class ETCar extends Car{ } //--------------------------------------代碼塊三--------------------------------------- public abstract class PCar extends Car { } //--------------------------------------代碼塊四--------------------------------------- public class RedETCar extends Car{@Overridepublic void move() {System.out.println("Red ETCar is running!");} } //--------------------------------------代碼塊五--------------------------------------- public class RedPCar extends PCar {@Overridepublic void move() {System.out.println("Red PCar is running!");} } //--------------------------------------代碼塊六--------------------------------------- public class WhiteETCar extends ETCar {@Overridepublic void move() {System.out.println("White ETCar is running!");} } //--------------------------------------代碼塊七--------------------------------------- public class WhitePCar extends PCar {@Overridepublic void move() {System.out.println("White PCar is running!");} } //--------------------------------------代碼塊八--------------------------------------- public class Main {public static void main(String[] args) {new RedETCar().move();new RedPCar().move();new WhitePCar().move();new WhiteETCar().move();} } 復(fù)制代碼

    結(jié)果:

    3.7.3 Good Code

    正確的方式是: 定義一個(gè)抽象基類汽車Car。汽車Car分為兩種,一種是油車PCar,一種是電動(dòng)車ETCar。

    因?yàn)槌橄蠡惼嘋ar合成復(fù)用了IColor接口對(duì)象,所以子類油車PCar和電動(dòng)車ETCar可以持有抽象基類Car的IColor接口對(duì)象。

    因?yàn)镮Color對(duì)象一個(gè)接口,接口有多種顏色: 白色、黑色、黃色、綠色、棕等等。

    如果每增加一種顏色,那么實(shí)現(xiàn)IColor接口即可,不需要像Bad Code通過(guò)繼承方式進(jìn)行類組合,不但解決了類爆炸的問(wèn)題,而且解決了繼承帶來(lái)的高耦合弊端。因此,在類組合問(wèn)題上,我們可以利用合成復(fù)用原則解決代碼冗余問(wèn)題。

    //--------------------------------------代碼塊一--------------------------------------- public interface IColor {String getName(); } //--------------------------------------代碼塊二--------------------------------------- public class RedColor implements IColor {@Overridepublic String getName() {return "Red";} } //--------------------------------------代碼塊三--------------------------------------- public class WhiteColor implements IColor {@Overridepublic String getName() {return "White";} } //--------------------------------------代碼塊四--------------------------------------- public abstract class Car {private IColor color;public abstract void move();public IColor getColor() {return color;}public Car setColor(IColor color) {this.color = color;return this;} } //--------------------------------------代碼塊五--------------------------------------- public class PCar extends Car {@Overridepublic void move() {System.out.println(getColor().getName() + " "+PCar.class.getSimpleName() +" is running!" );} } //--------------------------------------代碼塊六--------------------------------------- public class ETCar extends Car {@Overridepublic void move() {System.out.println(getColor().getName() + " "+PCar.class.getSimpleName() +" is running!" );} } //--------------------------------------代碼塊七--------------------------------------- public class Main {public static void main(String[] args) {PCar pCar = new PCar();ETCar etCar = new ETCar();RedColor redColor = new RedColor();WhiteColor whiteColor = new WhiteColor();pCar.setColor(redColor).move();pCar.setColor(whiteColor).move();etCar.setColor(redColor).move();etCar.setColor(whiteColor).move();} } 復(fù)制代碼

    結(jié)果:

    3.7.4 思考復(fù)盤

    組合和聚合到底有什么區(qū)別呢?

    聚合關(guān)系的類里有另外一個(gè)類作為參數(shù)。BirdGroup類被gc之后,bird類的引用依然建在。這就是聚合。

    public class BirdGroup{public Bird bird;public BirdGroup(Bird bird){this.bird = bird;} } 復(fù)制代碼

    組合關(guān)系的類里有另外一個(gè)類的實(shí)例化,如果Bird這個(gè)類被GC了,內(nèi)部的類的引用,隨之消失了,這就是組合。

    public class Bird{public Wings wings;public Bird(){wings = new Wings () ;} } 復(fù)制代碼

    合成復(fù)用原則的優(yōu)點(diǎn)

    使系統(tǒng)更加靈活,降低類與類之間的耦合度,一個(gè)類的變化對(duì)其他類造成的影響相對(duì)較小。

    合成復(fù)用原則的缺點(diǎn)

    破壞了包裝,同時(shí)包含的類的實(shí)現(xiàn)細(xì)節(jié)被隱藏。

    好了,七大設(shè)計(jì)原則到現(xiàn)在已經(jīng)說(shuō)完了,我們簡(jiǎn)單的總結(jié)一下:

    如果大家覺的上面表格比較復(fù)雜,那么用七句話總結(jié)就是:

    單一職責(zé)原則告訴我們實(shí)現(xiàn)類要職責(zé)單一;

    里氏替換原則告訴我們不要破壞繼承體系;

    依賴倒置原則告訴我們要面向接口編程;

    接口隔離原則告訴我們?cè)谠O(shè)計(jì)接口的時(shí)候要精簡(jiǎn)單一;

    最小知識(shí)原則告訴我們要降低耦合;

    合成復(fù)用原則告訴我們不要通過(guò)繼承方式實(shí)現(xiàn)功能和類組合;

    而開閉原則是總綱,告訴我們要對(duì)擴(kuò)展開放,對(duì)修改關(guān)閉。

    四、3大設(shè)計(jì)模式

    說(shuō)完七大設(shè)計(jì)原則,我們?cè)僬f(shuō)說(shuō)3大設(shè)計(jì)模式,設(shè)計(jì)模式一般分為三種,第一種是創(chuàng)建型模式,第二種是結(jié)構(gòu)型模式,第三種是行為型模式。

    當(dāng)我們關(guān)注類的對(duì)象,比如如何孵化出來(lái)類的對(duì)象?如何創(chuàng)建類的對(duì)象?如何new出來(lái)類的對(duì)象?如何維護(hù)類的對(duì)象關(guān)系?我們就需要使用到創(chuàng)建型模式。

    當(dāng)我們關(guān)注類與類之間的關(guān)系,如 A 跟 B 類組合或生產(chǎn)關(guān)系的時(shí)候。我們就需要使用到結(jié)構(gòu)型模式。

    當(dāng)我們關(guān)注類某一個(gè)方法功能的一個(gè)實(shí)現(xiàn),我們就需要使用到行為型模式。

    創(chuàng)建型模式、結(jié)構(gòu)型模式和行為型模式又分為23 種,由于篇幅有限,今天主要講解創(chuàng)建型模式的建造者設(shè)計(jì)模式,結(jié)構(gòu)型模式的適配器設(shè)計(jì)模式,行為型模式的策略設(shè)計(jì)模式和模板方法設(shè)計(jì)模式。剩余19種設(shè)計(jì)模式,小木箱將在后續(xù)文章進(jìn)行講解和梳理。

    4.1 創(chuàng)建型模式

    創(chuàng)建型模式本質(zhì)上是處理類的實(shí)例化,封裝了具體類的信息和隱藏了類的實(shí)例化過(guò)程。今天主要講解建造者設(shè)計(jì)模式

    4.1.1 建造者設(shè)計(jì)模式

    4.1.1.1 定義

    建造者模式所完成的內(nèi)容就是通過(guò)將多個(gè)簡(jiǎn)單對(duì)象通過(guò)一步步的組裝構(gòu)建出一個(gè)復(fù)雜對(duì)象的過(guò)程。

    建造者設(shè)計(jì)模式滿足了單一職責(zé)原則以及可復(fù)用的技術(shù)、建造者獨(dú)立、易擴(kuò)展、便于控制細(xì)節(jié)風(fēng)險(xiǎn)。

    但同時(shí)當(dāng)出現(xiàn)特別多的物料以及很多的組合后,類的不斷擴(kuò)展也會(huì)造成難以維護(hù)的問(wèn)題。

    建造者設(shè)計(jì)模式可以把重復(fù)的內(nèi)容抽象到數(shù)據(jù)庫(kù)中,按照需要配置。這樣就可以減少代碼中大量的重復(fù)。

    4.1.1.2 B站視頻

    《重學(xué)Java設(shè)計(jì)模式》第6章:建造者模式

    4.1.1.3 Bad Code

    這里我們模擬裝修公司對(duì)于設(shè)計(jì)出一些套餐裝修服務(wù)的場(chǎng)景。

    很多裝修公司都會(huì)給出自家的套餐服務(wù),一般有;歐式豪華、輕奢田園、現(xiàn)代簡(jiǎn)約等等,而這些套餐的后面是不同的商品的組合。例如;一級(jí)&二級(jí)吊頂、多樂(lè)士涂料、圣象地板、馬可波羅地磚等等,按照不同的套餐的價(jià)格選取不同的品牌組合,最終再按照裝修面積給出一個(gè)整體的報(bào)價(jià)。

    這里我們就模擬裝修公司想推出一些套餐裝修服務(wù),按照不同的價(jià)格設(shè)定品牌選擇組合,以達(dá)到使用建造者模式的過(guò)程。

    在模擬工程中提供了裝修中所需要的物料;ceilling(吊頂)、coat(涂料)、floor(地板)、tile(地磚),這么四項(xiàng)內(nèi)容。(實(shí)際的裝修物料要比這個(gè)多的多)

    4.1.1.3.1 代碼結(jié)構(gòu)

    • 物料接口: Matter

      • 物料接口提供了基本的信息,以保證所有的裝修材料都可以按照統(tǒng)一標(biāo)準(zhǔn)進(jìn)行獲取。
    public interface Matter {String scene(); // 場(chǎng)景;地板、地磚、涂料、吊頂String brand(); // 品牌String model(); // 型號(hào)BigDecimal price(); // 價(jià)格String desc(); // 描述} 復(fù)制代碼
    • 吊頂(ceiling)

      • 一級(jí)頂: LevelOneCeiling
    public class LevelOneCeiling implements Matter {public String scene() {return "吊頂";}public String brand() {return "裝修公司自帶";}public String model() {return "一級(jí)頂";}public BigDecimal price() {return new BigDecimal(260);}public String desc() {return "造型只做低一級(jí),只有一個(gè)層次的吊頂,一般離頂120-150mm";}} 復(fù)制代碼
    • 二級(jí)頂: LevelTwoCeiling
    public class LevelTwoCeiling implements Matter {public String scene() {return "吊頂";}public String brand() {return "裝修公司自帶";}public String model() {return "二級(jí)頂";}public BigDecimal price() {return new BigDecimal(850);}public String desc() {return "兩個(gè)層次的吊頂,二級(jí)吊頂高度一般就往下吊20cm,要是層高很高,也可增加每級(jí)的厚度";}} 復(fù)制代碼
    • 涂料(coat)

      • 多樂(lè)士: DuluxCoat
    public class DuluxCoat implements Matter {public String scene() {return "涂料";}public String brand() {return "多樂(lè)士(Dulux)";}public String model() {return "第二代";}public BigDecimal price() {return new BigDecimal(719);}public String desc() {return "多樂(lè)士是阿克蘇諾貝爾旗下的著名建筑裝飾油漆品牌,產(chǎn)品暢銷于全球100個(gè)國(guó)家,每年全球有5000萬(wàn)戶家庭使用多樂(lè)士油漆。";}} 復(fù)制代碼
    • 立邦: LiBangCoat
    public class LiBangCoat implements Matter {public String scene() {return "涂料";}public String brand() {return "立邦";}public String model() {return "默認(rèn)級(jí)別";}public BigDecimal price() {return new BigDecimal(650);}public String desc() {return "立邦始終以開發(fā)綠色產(chǎn)品、注重高科技、高品質(zhì)為目標(biāo),以技術(shù)力量不斷推進(jìn)科研和開發(fā),滿足消費(fèi)者需求。";}} 復(fù)制代碼
    • 地板(floor)

      • 德爾
    public class DerFloor implements Matter {public String scene() {return "地板";}public String brand() {return "德爾(Der)";}public String model() {return "A+";}public BigDecimal price() {return new BigDecimal(119);}public String desc() {return "DER德爾集團(tuán)是全球領(lǐng)先的專業(yè)木地板制造商,北京2008年奧運(yùn)會(huì)家裝和公裝地板供應(yīng)商";}} 復(fù)制代碼
    • 圣象
    public class ShengXiangFloor implements Matter {public String scene() {return "地板";}public String brand() {return "圣象";}public String model() {return "一級(jí)";}public BigDecimal price() {return new BigDecimal(318);}public String desc() {return "圣象地板是中國(guó)地板行業(yè)著名品牌。圣象地板擁有中國(guó)馳名商標(biāo)、中國(guó)名牌、國(guó)家免檢、中國(guó)環(huán)境標(biāo)志認(rèn)證等多項(xiàng)榮譽(yù)。";}} 復(fù)制代碼
    • 地磚(tile)
    public class DongPengTile implements Matter {public String scene() {return "地磚";}public String brand() {return "東鵬瓷磚";}public String model() {return "10001";}public BigDecimal price() {return new BigDecimal(102);}public String desc() {return "東鵬瓷磚以品質(zhì)鑄就品牌,科技推動(dòng)品牌,口碑傳播品牌為宗旨,2014年品牌價(jià)值132.35億元,位列建陶行業(yè)榜首。";}} 復(fù)制代碼
    • 馬可波羅
    public class MarcoPoloTile implements Matter {public String scene() {return "地磚";}public String brand() {return "馬可波羅(MARCO POLO)";}public String model() {return "缺省";}public BigDecimal price() {return new BigDecimal(140);}public String desc() {return "“馬可波羅”品牌誕生于1996年,作為國(guó)內(nèi)最早品牌化的建陶品牌,以“文化陶瓷”占領(lǐng)市場(chǎng),享有“仿古磚至尊”的美譽(yù)。";}} 復(fù)制代碼

    以上就是本次裝修公司所提供的裝修配置單,接下我們會(huì)通過(guò)案例去使用不同的物料組合出不同的套餐服務(wù)。

    public class DecorationPackageController {public String getMatterList(BigDecimal area, Integer level) {List<Matter> list = new ArrayList<Matter>(); // 裝修清單BigDecimal price = BigDecimal.ZERO; // 裝修價(jià)格// 豪華歐式if (1 == level) {LevelTwoCeiling levelTwoCeiling = new LevelTwoCeiling(); // 吊頂,二級(jí)頂DuluxCoat duluxCoat = new DuluxCoat(); // 涂料,多樂(lè)士ShengXiangFloor shengXiangFloor = new ShengXiangFloor(); // 地板,圣象list.add(levelTwoCeiling);list.add(duluxCoat);list.add(shengXiangFloor);price = price.add(area.multiply(new BigDecimal("0.2")).multiply(levelTwoCeiling.price()));price = price.add(area.multiply(new BigDecimal("1.4")).multiply(duluxCoat.price()));price = price.add(area.multiply(shengXiangFloor.price()));}// 輕奢田園if (2 == level) {LevelTwoCeiling levelTwoCeiling = new LevelTwoCeiling(); // 吊頂,二級(jí)頂LiBangCoat liBangCoat = new LiBangCoat(); // 涂料,立邦MarcoPoloTile marcoPoloTile = new MarcoPoloTile(); // 地磚,馬可波羅list.add(levelTwoCeiling);list.add(liBangCoat);list.add(marcoPoloTile);price = price.add(area.multiply(new BigDecimal("0.2")).multiply(levelTwoCeiling.price()));price = price.add(area.multiply(new BigDecimal("1.4")).multiply(liBangCoat.price()));price = price.add(area.multiply(marcoPoloTile.price()));}// 現(xiàn)代簡(jiǎn)約if (3 == level) {LevelOneCeiling levelOneCeiling = new LevelOneCeiling(); // 吊頂,二級(jí)頂LiBangCoat liBangCoat = new LiBangCoat(); // 涂料,立邦DongPengTile dongPengTile = new DongPengTile(); // 地磚,東鵬list.add(levelOneCeiling);list.add(liBangCoat);list.add(dongPengTile);price = price.add(area.multiply(new BigDecimal("0.2")).multiply(levelOneCeiling.price()));price = price.add(area.multiply(new BigDecimal("1.4")).multiply(liBangCoat.price()));price = price.add(area.multiply(dongPengTile.price()));}StringBuilder detail = new StringBuilder("\r\n-------------------------------------------------------\r\n" +"裝修清單" + "\r\n" +"套餐等級(jí):" + level + "\r\n" +"套餐價(jià)格:" + price.setScale(2, BigDecimal.ROUND_HALF_UP) + " 元\r\n" +"房屋面積:" + area.doubleValue() + " 平米\r\n" +"材料清單:\r\n");for (Matter matter: list) {detail.append(matter.scene()).append(":").append(matter.brand()).append("、").append(matter.model()).append("、平米價(jià)格:").append(matter.price()).append(" 元。\n");}return detail.toString();}} 復(fù)制代碼
    • 測(cè)試入口: Main
    public class Main {public static void main(String[] args) {DecorationPackageController decoration = new DecorationPackageController();// 豪華歐式System.out.println(decoration.getMatterList(new BigDecimal("132.52"),1));// 輕奢田園System.out.println(decoration.getMatterList(new BigDecimal("98.25"),2));// 現(xiàn)代簡(jiǎn)約System.out.println(decoration.getMatterList(new BigDecimal("85.43"),3));} } 復(fù)制代碼

    總結(jié):

  • 首先這段代碼所要解決的問(wèn)題就是接收入?yún)?#xff1b;裝修面積(area)、裝修等級(jí)(level),根據(jù)不同類型的裝修等級(jí)選擇不同的材料。
  • 其次在實(shí)現(xiàn)過(guò)程中可以看到每一段if塊里,都包含著不同的材料(吊頂,二級(jí)頂、涂料,立邦、地磚,馬可波羅),最終生成裝修清單和裝修成本。
  • 最后提供獲取裝修詳細(xì)信息的方法,返回給調(diào)用方,用于知道裝修清單。
  • 4.1.1.3.2?輸出結(jié)果

    ------------------------------------------------------- 裝修清單 套餐等級(jí):1 套餐價(jià)格:198064.39 元 房屋面積:132.52 平米 材料清單: 吊頂:裝修公司自帶、二級(jí)頂、平米價(jià)格:850 元。 涂料:多樂(lè)士(Dulux)、第二代、平米價(jià)格:719 元。 地板:圣象、一級(jí)、平米價(jià)格:318 元。------------------------------------------------------- 裝修清單 套餐等級(jí):2 套餐價(jià)格:119865.00 元 房屋面積:98.25 平米 材料清單: 吊頂:裝修公司自帶、二級(jí)頂、平米價(jià)格:850 元。 涂料:立邦、默認(rèn)級(jí)別、平米價(jià)格:650 元。 地磚:馬可波羅(MARCO POLO)、缺省、平米價(jià)格:140 元。------------------------------------------------------- 裝修清單 套餐等級(jí):3 套餐價(jià)格:90897.52 元 房屋面積:85.43 平米 材料清單: 吊頂:裝修公司自帶、一級(jí)頂、平米價(jià)格:260 元。 涂料:立邦、默認(rèn)級(jí)別、平米價(jià)格:650 元。 地磚:東鵬瓷磚、10001、平米價(jià)格:102 元。 復(fù)制代碼

    4.1.1.4 Good Code

    工程結(jié)構(gòu)

    ├── Builder.java ├── DecorationPackageMenu.java ├── IMenu.java ├── Main.java ├── ceiling │?? ├── LevelOneCeiling.java │?? ├── LevelTwoCeiling.java │?? └── Matter.java ├── coat │?? ├── DuluxCoat.java │?? └── LiBangCoat.java ├── floor │?? ├── DerFloor.java │?? └── ShengXiangFloor.java └── tile ├── DongPengTile.java └── MarcoPoloTile.java 復(fù)制代碼

    建造者模型結(jié)構(gòu)

    工程中有三個(gè)核心類和一個(gè)測(cè)試類,核心類是建造者模式的具體實(shí)現(xiàn)。與ifelse實(shí)現(xiàn)方式相比,多出來(lái)了兩個(gè)二外的類。具體功能如下;

    • Builder,建造者類具體的各種組裝由此類實(shí)現(xiàn)。
    • DecorationPackageMenu,是IMenu接口的實(shí)現(xiàn)類,主要是承載建造過(guò)程中的填充器。相當(dāng)于這是一套承載物料和創(chuàng)建者中間銜接的內(nèi)容。

    好,那么接下來(lái)會(huì)分別講解幾個(gè)類的具體實(shí)現(xiàn)

    定義裝修包接口

    public interface IMenu {IMenu appendCeiling(Matter matter); // 吊頂IMenu appendCoat(Matter matter); // 涂料IMenu appendFloor(Matter matter); // 地板IMenu appendTile(Matter matter); // 地磚String getDetail(); // 明細(xì) } 復(fù)制代碼
    • 接口類中定義了填充各項(xiàng)物料的方法;吊頂、涂料、地板、地磚,以及最終提供獲取全部明細(xì)的方法。

    裝修包實(shí)現(xiàn)

    public class DecorationPackageMenu implements IMenu {private List<Matter> list = new ArrayList<Matter>(); // 裝修清單private BigDecimal price = BigDecimal.ZERO; // 裝修價(jià)格private BigDecimal area; // 面積private String grade; // 裝修等級(jí);豪華歐式、輕奢田園、現(xiàn)代簡(jiǎn)約private DecorationPackageMenu() {}public DecorationPackageMenu(Double area, String grade) {this.area = new BigDecimal(area);this.grade = grade;}public IMenu appendCeiling(Matter matter) {list.add(matter);price = price.add(area.multiply(new BigDecimal("0.2")).multiply(matter.price()));return this;}public IMenu appendCoat(Matter matter) {list.add(matter);price = price.add(area.multiply(new BigDecimal("1.4")).multiply(matter.price()));return this;}public IMenu appendFloor(Matter matter) {list.add(matter);price = price.add(area.multiply(matter.price()));return this;}public IMenu appendTile(Matter matter) {list.add(matter);price = price.add(area.multiply(matter.price()));return this;}public String getDetail() {StringBuilder detail = new StringBuilder("\r\n-------------------------------------------------------\r\n" +"裝修清單" + "\r\n" +"套餐等級(jí):" + grade + "\r\n" +"套餐價(jià)格:" + price.setScale(2, BigDecimal.ROUND_HALF_UP) + " 元\r\n" +"房屋面積:" + area.doubleValue() + " 平米\r\n" +"材料清單:\r\n");for (Matter matter: list) {detail.append(matter.scene()).append(":").append(matter.brand()).append("、").append(matter.model()).append("、平米價(jià)格:").append(matter.price()).append(" 元。\n");}return detail.toString();}} 復(fù)制代碼
    • 裝修包的實(shí)現(xiàn)中每一個(gè)方法都會(huì)了?this,也就可以非常方便的用于連續(xù)填充各項(xiàng)物料。
    • 同時(shí)在填充時(shí)也會(huì)根據(jù)物料計(jì)算平米數(shù)下的報(bào)價(jià),吊頂和涂料按照平米數(shù)適量乘以常數(shù)計(jì)算。
    • 最后同樣提供了統(tǒng)一的獲取裝修清單的明細(xì)方法。

    建造者方法

    public class Builder {public IMenu levelOne(Double area) {return new DecorationPackageMenu(area, "豪華歐式").appendCeiling(new LevelTwoCeiling()) // 吊頂,二級(jí)頂.appendCoat(new DuluxCoat()) // 涂料,多樂(lè)士.appendFloor(new ShengXiangFloor()); // 地板,圣象}public IMenu levelTwo(Double area){return new DecorationPackageMenu(area, "輕奢田園").appendCeiling(new LevelTwoCeiling()) // 吊頂,二級(jí)頂.appendCoat(new LiBangCoat()) // 涂料,立邦.appendTile(new MarcoPoloTile()); // 地磚,馬可波羅}public IMenu levelThree(Double area){return new DecorationPackageMenu(area, "現(xiàn)代簡(jiǎn)約").appendCeiling(new LevelOneCeiling()) // 吊頂,二級(jí)頂.appendCoat(new LiBangCoat()) // 涂料,立邦.appendTile(new DongPengTile()); // 地磚,東鵬}} 復(fù)制代碼

    測(cè)試方法:

    @Test public void test_Builder(){Builder builder = new Builder();// 豪華歐式System.out.println(builder.levelOne(132.52D).getDetail());// 輕奢田園System.out.println(builder.levelTwo(98.25D).getDetail());// 現(xiàn)代簡(jiǎn)約System.out.println(builder.levelThree(85.43D).getDetail()); } 復(fù)制代碼

    結(jié)果:

    ------------------------------------------------------- 裝修清單 套餐等級(jí):豪華歐式 套餐價(jià)格:198064.39 元 房屋面積:132.52 平米 材料清單: 吊頂:裝修公司自帶、二級(jí)頂、平米價(jià)格:850 元。 涂料:多樂(lè)士(Dulux)、第二代、平米價(jià)格:719 元。 地板:圣象、一級(jí)、平米價(jià)格:318 元。------------------------------------------------------- 裝修清單 套餐等級(jí):輕奢田園 套餐價(jià)格:119865.00 元 房屋面積:98.25 平米 材料清單: 吊頂:裝修公司自帶、二級(jí)頂、平米價(jià)格:850 元。 涂料:立邦、默認(rèn)級(jí)別、平米價(jià)格:650 元。 地磚:馬可波羅(MARCO POLO)、缺省、平米價(jià)格:140 元。------------------------------------------------------- 裝修清單 套餐等級(jí):現(xiàn)代簡(jiǎn)約 套餐價(jià)格:90897.52 元 房屋面積:85.43 平米 材料清單: 吊頂:裝修公司自帶、一級(jí)頂、平米價(jià)格:260 元。 涂料:立邦、默認(rèn)級(jí)別、平米價(jià)格:650 元。 地磚:東鵬瓷磚、10001、平米價(jià)格:102 元 復(fù)制代碼
    • 測(cè)試結(jié)果是一樣的,調(diào)用方式也基本類似。但是目前的代碼結(jié)構(gòu)卻可以讓你很方便的很有調(diào)理的進(jìn)行擴(kuò)展業(yè)務(wù)開發(fā)。而不是像以往一樣把所有代碼都寫到ifelse里面。

    4.1.1.5 Source Code

    建造者不拘泥于形式,建造者模式用于創(chuàng)建一個(gè)復(fù)雜對(duì)象。在android中,Dialog就用到了建造者模式,第三方庫(kù)的okhttp、Retrofit等

    public class Dialog {String title;boolean mCancelable = false;Dialog(String title,boolean mCanclable){this.title = title;this.mCancelable = mCanclable;}public void show() {System.out.print("show");}static class Builder{String title;boolean mCancelable = false;public Builder setCancelable(boolean flag) {mCancelable = flag;return this;}public Builder setTitle(String title) {this.title = title;return this;}public Dialog build(){return new Dialog(this.title,this.mCancelable);}} } 復(fù)制代碼

    4.1.1.6 注意事項(xiàng)

    優(yōu)點(diǎn):

    客戶端不比知道產(chǎn)品內(nèi)部細(xì)節(jié),將產(chǎn)品本身與產(chǎn)品創(chuàng)建過(guò)程解耦,使得相同的創(chuàng)建過(guò)程可以創(chuàng)建不同的產(chǎn)品對(duì)象可以更加精細(xì)地控制產(chǎn)品的創(chuàng)建過(guò)程,將復(fù)雜對(duì)象分門別類抽出不同的類別來(lái),使得開發(fā)者可以更加方便地得到想要的產(chǎn)品

    缺點(diǎn):

    產(chǎn)品屬性之間差異很大且屬性沒(méi)有默認(rèn)值可以指定,這種情況是沒(méi)法使用建造者模式的,我們可以試想,一個(gè)對(duì)象20個(gè)屬性,彼此之間毫無(wú)關(guān)聯(lián)且每個(gè)都需要手動(dòng)指定,那么很顯然,即使使用了建造者模式也是毫無(wú)作用

    4.2 結(jié)構(gòu)型模式

    創(chuàng)建型模式本質(zhì)上是處理類或?qū)ο蟮慕M合,常見的結(jié)構(gòu)模型有類結(jié)構(gòu)型和對(duì)象結(jié)構(gòu)型。今天主要講解適配器設(shè)計(jì)模式

    4.2.1 適配器設(shè)計(jì)模式

    4.1.1.1 定義

    適配器模式把一個(gè)類的接口變換成客戶端所期待的另一種接口,從而使原本因接口不匹配而無(wú)法在一起工作的兩個(gè)類能夠在一起工作,是作為兩個(gè)不兼容的接口之間的橋梁。

    這種類型的設(shè)計(jì)模式屬于結(jié)構(gòu)型模式,它結(jié)合了兩個(gè)獨(dú)立接口的功能,適配器分為類適配器和對(duì)象適配器.

    主要解決在軟件系統(tǒng)中,常常要將一些"現(xiàn)存的對(duì)象"放到新的環(huán)境中,而新環(huán)境要求的接口是現(xiàn)對(duì)象不能滿足的;

    4.1.1.2 UML圖例

    4.1.1.3 類適配器

    類適配器是通過(guò)類的繼承來(lái)實(shí)現(xiàn)的。Adpater直接繼承了Target和Adaptee中的所有方法,并進(jìn)行改寫,從而實(shí)現(xiàn)了Target中的方法。

    類適配器的缺點(diǎn)就是必須實(shí)現(xiàn)Target和Adaptee中的方法,由于Java不支持多繼承,所以通常將Target設(shè)計(jì)成接口,Adapter繼承自Adaptee然后實(shí)現(xiàn)Target接口。使用類適配器的方式來(lái)實(shí)現(xiàn)一下上邊的用雄蜂來(lái)冒充鴨子。

    我們可以看到下面的案例雄蜂(Drone)具有蜂鳴聲(beep)、轉(zhuǎn)子旋轉(zhuǎn)(spin_rotors)和起飛(take_off)行為,鴨子Duck具有嘎嘎叫(quack)和飛(fly)行為

    那么如何找到一個(gè)適配器讓雄蜂(Drone)的蜂鳴聲beep和鴨子(Duck)的嘎嘎叫(quack)適配呢

    又如何找到一個(gè)適配器讓鴨子(鴨子)飛(fly)和雄蜂(Drone)的轉(zhuǎn)子旋轉(zhuǎn)(spin_rotors)、起飛(take_off)適配呢?

    很顯然雄蜂適配器(DroneAdapter)嘎嘎叫(quack)可以適配雄蜂(Drone)蜂鳴聲(beep)

    雄蜂適配器(DroneAdapter)嘎嘎叫(fly)也可以適配雄蜂(Drone)轉(zhuǎn)子旋轉(zhuǎn)(spin_rotors)和起飛(take_off)

    //--------------------------------------代碼塊一--------------------------------------- public interface Drone {void beep();void spin_rotors();void take_off(); } //--------------------------------------代碼塊二--------------------------------------- public class SuperDrone implements Drone {public void beep() {System.out.println("Beep beep beep");}public void spin_rotors() {System.out.println("Rotors are spinning");}public void take_off() {System.out.println("Taking off");} } //--------------------------------------代碼塊三--------------------------------------- public interface Duck {public void quack();public void fly(); } //--------------------------------------代碼塊四--------------------------------------- public class DroneAdapter implements Duck {Drone drone;public DroneAdapter(Drone drone) {this.drone = drone;}public void quack() {drone.beep();}public void fly() {drone.spin_rotors();drone.take_off();} } //--------------------------------------代碼塊五--------------------------------------- public class DuckTestDrive {public static void main(String[] args) {Drone drone = new SuperDrone();Duck droneAdapter = new DroneAdapter(drone);droneAdapter.quack();droneAdapter.fly();} } 復(fù)制代碼

    結(jié)果:

    4.1.1.4 對(duì)象適配器

    對(duì)象適配器是使用組合的方法,在Adapter中會(huì)保留一個(gè)原對(duì)象(Adaptee)的引用,適配器的實(shí)現(xiàn)就是講Target中的方法委派給Adaptee對(duì)象來(lái)做,用Adaptee中的方法實(shí)現(xiàn)Target中的方法

    對(duì)象適配器的好處就是,Adpater只需要實(shí)現(xiàn)Target中的方法就好啦。現(xiàn)在我們通過(guò)一個(gè)用火雞冒充鴨子的例子來(lái)看看如何使用適配器模式。

    火雞(Turkey)具備火雞叫(gobble)和飛(fly)行為,鴨子(Duck)具備嘎嘎叫(quack)和飛(fly)的行為,找一個(gè)火雞適配器(TurkeyAdapter)讓鴨子(Duck)的嘎嘎叫(quack)適配火雞(Turkey)的火雞叫(gobble).讓鴨子(Duck)的飛(fly)適配火雞(Turkey)的飛(fly),只要把火雞(Turkey)的對(duì)象傳給火雞適配器(TurkeyAdapter)即可.不改變野火雞(WildTurkey)火雞叫(gobble)和飛(fly)的行為.同時(shí),不改變綠頭鴨(MallardDuck)的嘎嘎叫(quack) 和飛(fly)的行為.

    //--------------------------------------代碼塊一--------------------------------------- public interface Duck {public void quack();public void fly(); } //--------------------------------------代碼塊二--------------------------------------- public interface Turkey {public void gobble();public void fly(); } //--------------------------------------代碼塊三--------------------------------------- public class TurkeyAdapter implements Duck {Turkey turkey;public TurkeyAdapter(Turkey turkey) {this.turkey = turkey;}public void quack() {turkey.gobble();}public void fly() {turkey.fly();} } //--------------------------------------代碼塊四--------------------------------------- public class WildTurkey implements Turkey {public void gobble() {System.out.println("Gobble gobble");}public void fly() {System.out.println("I'm flying a short distance");} } //--------------------------------------代碼塊五--------------------------------------- public class MallardDuck implements Duck {public void quack() {System.out.println("Quack");}public void fly() {System.out.println("I'm flying");} } //--------------------------------------代碼塊六--------------------------------------- public class DuckTestDrive {public static void main(String[] args) {Duck duck = new MallardDuck();Turkey turkey = new WildTurkey();Duck turkeyAdapter = new TurkeyAdapter(turkey);System.out.println("The Turkey says...");turkey.gobble();turkey.fly();System.out.println("\nThe Duck says...");testDuck(duck);System.out.println("\nThe TurkeyAdapter says...");testDuck(turkeyAdapter);}static void testDuck(Duck duck) {duck.quack();duck.fly();} } 復(fù)制代碼

    鴨子和火雞有相似之處,他們都會(huì)飛,雖然飛的不遠(yuǎn),他們不太一樣的地方就是叫聲不太一樣,現(xiàn)在我們有一個(gè)火雞的類,有鴨子的抽象類也就是接口。

    我們的適配器繼承自鴨子類并且保留了火雞的引用,重寫鴨子的飛和叫的方法,但是是委托給火雞的方法來(lái)實(shí)現(xiàn)的。在客戶端中,我們給適配器傳遞一個(gè)火雞的對(duì)象,就可以把它當(dāng)做鴨子來(lái)使用了。

    結(jié)果:

    4.1.1.5 Source Code

    適配器模式可以用繼承實(shí)現(xiàn),這里沒(méi)有更高的抽象,當(dāng)然也可以把Adapter的內(nèi)容抽象出去,僅僅演示,ListView、GridView適配了Adapter類。

    //定義適配器類 public class Adapter {public void getView(int i){System.out.println("給出View"+i);} } //ListView 繼承了Adapter public class ListView extends Adapter{public void show(){System.out.print("循環(huán)顯示View");for(int i=0;i<3;i++){getView(i);}} } //GridView繼承了Adapter public class GridView extends Adapter{public void show(){...getView(i);} } 復(fù)制代碼

    在android中,ListView、RecyclerView都是用了適配器模式,ListView適配了Adapter,ListView只管ItemView,不管具體怎么展示,Adapter只管展示。就像讀卡器,讀卡器作為內(nèi)存和電腦之間的適配器。

    4.1.1.6 注意事項(xiàng)

    適配器模式的優(yōu)點(diǎn):

  • 將目標(biāo)類和適配者類解耦,通過(guò)引入一個(gè)適配器類來(lái)重用現(xiàn)有的適配者類,而無(wú)須修改原有代碼。

  • 增加了類的透明性和復(fù)用性,將具體的實(shí)現(xiàn)封裝在適配者類中,對(duì)于客戶端類來(lái)說(shuō)是透明的,而且提高了適配者的復(fù)用性。

  • 靈活性和擴(kuò)展性都非常好,通過(guò)使用配置文件,可以很方便地更換適配器,也可以在不修改原有代碼的基礎(chǔ)上增加新的適配器類,完全符合“開閉原則”。

  • 適配器模式的缺點(diǎn):

  • 過(guò)多地使用適配器,會(huì)讓系統(tǒng)非常零亂,不易整體進(jìn)行把握。比如,明明看到調(diào)用的是 A 接口,其實(shí)內(nèi)部被適配成了 B 接口的實(shí)現(xiàn),一個(gè)系統(tǒng)如果太多出現(xiàn)這種情況,無(wú)異于一場(chǎng)災(zāi)難。因此如果不是很有必要,可以不使用適配器,而是直接對(duì)系統(tǒng)進(jìn)行重構(gòu)。

  • 由于 JAVA 至多繼承一個(gè)類,所以至多只能適配一個(gè)適配者類,而且目標(biāo)類必須是抽象類。

  • 一次最多只能適配一個(gè)適配者類,不能同時(shí)適配多個(gè)適配者。

  • 目標(biāo)抽象類只能為接口,不能為類,其使用有一定的局限性;

  • 適配器模式的使用時(shí)機(jī):

  • 在實(shí)際的開發(fā)過(guò)程中,一個(gè)接口有大量的方法,但是對(duì)應(yīng)的不同類只需要關(guān)注部分方法,其他無(wú)關(guān)的方法全都實(shí)現(xiàn)過(guò)于繁瑣,尤其是涉及的實(shí)現(xiàn)類過(guò)多的情況。

  • 想要建立一個(gè)可以重復(fù)使用的類,用于與一些彼此之間沒(méi)有太大關(guān)聯(lián)的一些類,包括一些可能在將來(lái)引進(jìn)的類一起工作。

  • 如: 現(xiàn)有一個(gè)需要的目標(biāo)接口對(duì)象 Target,定義了大量相關(guān)的方法。但是在實(shí)際使用過(guò)程只需分別關(guān)注其中部分方法,而不是全部實(shí)現(xiàn)。在此場(chǎng)景中:被依賴的目標(biāo)對(duì)象TargetObj、適配器Adapter、客戶端Client等

    // 目標(biāo)對(duì)象:定義了大量的相關(guān)方法 public interface TargetObj {void operation1();void operation2();void operation3();void operation4();void operation5(); }// 適配器:將目標(biāo)接口定義的方法全部做默認(rèn)實(shí)現(xiàn) public abstract class Adapter implements TargetObj {void operation1(){}void operation2(){}void operation3(){}void operation4(){}void operation5(){} }// 客戶端:采用匿名內(nèi)部類的方式實(shí)現(xiàn)需要的接口即可完成適配 public class Client {public static void main(String[] args) {Adapter adapter1 = new Adapter() {@Overridepublic void operation3() {// 僅僅實(shí)現(xiàn)需要關(guān)注的方法即可System.out.println("operation3")}}Adapter adapter2 = new Adapter() {@Overridepublic void operation5() {// 僅僅實(shí)現(xiàn)需要關(guān)注的方法即可System.out.println("operation5")}}adapter1.operation3();adapter2.operation5();}} 復(fù)制代碼

    4.3 行為型模式

    4.3.1 策略設(shè)計(jì)模式

    4.1.1.1 定義

    策略模式定義是一系列封裝起來(lái)的一種算法,讓算法與算法之間可以相互替換。策略模式把算法委托于使用者,策略模式可以獨(dú)立變化。

    比如我們要去某個(gè)地方,會(huì)根據(jù)距離的不同(或者是根據(jù)手頭經(jīng)濟(jì)狀況)來(lái)選擇不同的出行方式(共享單車、坐公交、滴滴打車等等),這些出行方式即不同的策略。

    再比如活動(dòng)促銷,打 9 折、打 3 折、打 7 折還是打 8 折?涉及具體的策略選擇時(shí)候,讓使用者選擇,使用者只關(guān)心對(duì)算法的封裝,我怎么樣去實(shí)現(xiàn)算法。使用者不需要管。下面我們就用策略設(shè)計(jì)模式實(shí)現(xiàn)一個(gè)圖書購(gòu)買系統(tǒng).

    4.1.2.2 Code Case

    在一個(gè)圖書購(gòu)買系統(tǒng)中,主要由一些幾種不同的折扣:

    折扣一(NoDiscountStrategy):對(duì)有些圖書沒(méi)有折扣。折扣算法對(duì)象返還0作為折扣值。

    折扣二(FlatRateStrategy):對(duì)有些圖書提供一個(gè)固定量值為1元的折扣。

    折扣三(PercentageStrategy):對(duì)有些圖書提供一個(gè)百分比的折扣,比如本書價(jià)格為 20元,折扣百分比為7%,那么折扣值就是20×7%=1.4(元)。

    //--------------------------------------代碼塊一--------------------------------------- public class Book {private String name;private DiscountStrategy strategy;public Book(String name, DiscountStrategy strategy) {this.name = name;this.strategy = strategy;}public void setStrategy(DiscountStrategy strategy) {this.strategy = strategy;}public void getDiscount(){System.out.println("book name:"+ name + " ,the discount algorithm is: "+ strategy.getClass().getSimpleName()+",the discounted price is: " + strategy.calcDiscount());} } //--------------------------------------代碼塊二--------------------------------------- public abstract class DiscountStrategy {private double price = 0;private int copies;public DiscountStrategy() {}public DiscountStrategy(double price, int copies) {this.price = price;this.copies = copies;}abstract double calcDiscount();public double getPrice() {return price;}public int getCopies() {return copies;} } //--------------------------------------代碼塊三--------------------------------------- public class FlatRateStrategy extends DiscountStrategy{private int discountPrice;public FlatRateStrategy(double price, int copies) {super(price,copies);}public void setDiscountPrice(int discountPrice) {this.discountPrice = discountPrice;}@Overridedouble calcDiscount() {return discountPrice * getCopies();} } //--------------------------------------代碼塊四--------------------------------------- public class NoDiscountStrategy extends DiscountStrategy{@Overridedouble calcDiscount() {return 0;} } //--------------------------------------代碼塊五--------------------------------------- public class PercentageStrategy extends DiscountStrategy{private double discountPercent;public PercentageStrategy(double price, int copies) {super(price, copies);}public void setDiscountPercent(double discountPercent) {this.discountPercent = discountPercent;}@Overridedouble calcDiscount() {return getCopies() * getPrice() * discountPercent;} } //--------------------------------------代碼塊六--------------------------------------- public class Client {public static void main(String[] args) {Book book1 = new Book("java design pattern", new NoDiscountStrategy());book1.getDiscount();FlatRateStrategy rateStrategy = new FlatRateStrategy(23.0, 5);rateStrategy.setDiscountPrice(1);Book book2 = new Book("java design pattern",rateStrategy);book2.getDiscount();System.out.println("Revise《java design pattern》discount algorithm\n:");PercentageStrategy percentageStrategy = new PercentageStrategy(23, 5);percentageStrategy.setDiscountPercent(0.07);book2.setStrategy(percentageStrategy);book2.getDiscount();} } 復(fù)制代碼

    結(jié)果:

    4.1.2.3 Android Code

    Android中RecyclerView的例子,我們給RecyclerView選擇布局方式的時(shí)候,就是選擇的策略模式

    //假如RecyclerView 這樣寫 public class RecyclerView {private Layout layout;public void setLayout(Layout layout) {this.layout = layout;if(layout == "橫著"){}else if(layout == "豎著"){}else if(layout=="格子"){}else{} this.layout.doLayout();} } //這樣寫if就很多了 //排列的方式 public interface Layout {void doLayout(); } //豎著排列 public class LinearLayout implements Layout{@Overridepublic void doLayout() {System.out.println("LinearLayout");} } //網(wǎng)格排列 public class GridLayout implements Layout{@Overridepublic void doLayout() {System.out.println("GridLayout");} } public class RecyclerView {private Layout layout;public void setLayout(Layout layout) {this.layout = layout;this.layout.doLayout();} } 復(fù)制代碼

    當(dāng)然Android的源碼里面動(dòng)畫時(shí)間插值器,用的也是策略設(shè)計(jì)模式,代碼就不貼了,大家可以結(jié)合源碼和Android設(shè)計(jì)模式之策略模式在項(xiàng)目中的實(shí)際使用總結(jié)文章中的UML圖進(jìn)行學(xué)習(xí).

    4.1.2.4 注意事項(xiàng)

    為什么要用策略設(shè)計(jì)模式?

    比如我們有微信支付,有支付寶支付,還有銀聯(lián)支付和招商支付。如果邏輯都通過(guò) if else 實(shí)現(xiàn),那么 if-else 塊中的代碼量比較大時(shí)候,后續(xù)代碼的擴(kuò)展和維護(hù)就會(huì)逐漸變得非常困難且容易出錯(cuò),就算使用Switch也同樣違反了:

    if (微信支付) {// 邏輯1 } else if (支付寶支付) {// 邏輯2 } else if (銀聯(lián)支付) {// 邏輯3 } else if(招商支付){// 邏輯4 }else{ // 邏輯5 } 復(fù)制代碼

    單一職責(zé)原則(一個(gè)類應(yīng)該只有一個(gè)發(fā)生變化的原因):因?yàn)橹笮薷娜魏我粋€(gè)邏輯,當(dāng)前類都會(huì)被修改

    開閉原則(對(duì)擴(kuò)展開放,對(duì)修改關(guān)閉):如果此時(shí)需要添加(刪除)某個(gè)邏輯,那么不可避免的要修改原來(lái)的代碼

    什么時(shí)候使用策略設(shè)計(jì)模式?

  • 如果在一個(gè)系統(tǒng)里面有許多類,它們之間的區(qū)別僅在于它們的行為,那么使用策略模式可以動(dòng)態(tài)地讓一個(gè)對(duì)象在許多行為中選擇一種行為。 一個(gè)系統(tǒng)需要?jiǎng)討B(tài)地在幾種算法中選擇一種。

  • 如果一個(gè)對(duì)象有很多的行為,如果不用恰當(dāng)?shù)哪J?#xff0c;這些行為就只好使用多重的條件選擇語(yǔ)句來(lái)實(shí)現(xiàn)。

  • 不希望客戶端知道復(fù)雜的、與算法相關(guān)的數(shù)據(jù)結(jié)構(gòu),在具體策略類中封裝算法和相關(guān)的數(shù)據(jù)結(jié)構(gòu),提高算法的保密性與安全性。

  • 策略模式的優(yōu)缺點(diǎn)是什么?

    優(yōu)點(diǎn):

    • 策略模式提供了對(duì)“開閉原則”的完美支持,用戶可以在不修改原有系統(tǒng)的基礎(chǔ)上選擇算法或行為,也可以靈活地增加新的算法或行為。
    • 策略模式提供了管理相關(guān)的算法族的辦法。
    • 策略模式提供了可以替換繼承關(guān)系的辦法。
    • 使用策略模式可以避免使用多重條件轉(zhuǎn)移語(yǔ)句。

    缺點(diǎn):

    • 客戶端必須知道所有的策略類,并自行決定使用哪一個(gè)策略類。

    • 策略模式將造成產(chǎn)生很多策略類,可以通過(guò)使用享元模式在一定程度上減少對(duì)象的數(shù)量。

    4.3.2 模板方法設(shè)計(jì)模式

    4.3.2.1 定義

    模版模式是說(shuō)對(duì)一個(gè)執(zhí)行過(guò)程進(jìn)行抽象分解,通過(guò)骨架和擴(kuò)展方法完成一個(gè)標(biāo)準(zhǔn)的主體邏輯和擴(kuò)展。我們很多時(shí)候,做監(jiān)控平臺(tái)也都是這樣的:對(duì)過(guò)程進(jìn)行標(biāo)準(zhǔn)化,對(duì)變化進(jìn)行定義,形成一個(gè)平臺(tái)邏輯和業(yè)務(wù)擴(kuò)展,完成一個(gè)產(chǎn)品模版。

    4.3.2.2?UML?圖例

    通過(guò)以下AbstractClass模板類我們可以看出來(lái),PrivitiveOperation1()和PrivitiveOperation2()全部封裝在TemplateMethod()抽象方法里面,TemplateMethod()抽象方法父類控制執(zhí)行順序,子類負(fù)責(zé)實(shí)現(xiàn)即可。通過(guò)封裝不變部分,擴(kuò)展可變部分和提取公共部分代碼,便于維護(hù)和可拓展性。

    提出問(wèn)題

    小木箱準(zhǔn)備煮茶和煮咖啡,煮茶的步驟有燒水、泡茶、加檸檬、倒水四個(gè)步驟,而煮咖啡的步驟有燒水、過(guò)濾咖啡、倒水、加牛奶四個(gè)步驟,請(qǐng)?jiān)诳刂婆_(tái)打印煮茶和煮咖啡的執(zhí)行流程。

    分析問(wèn)題

    煮茶和煮咖啡的步驟中燒水和倒水動(dòng)作是重復(fù)的,能不能抽取成模板方法呢?

    解決問(wèn)題

    可以參考UML圖例、Good Code、Bad Code和模板方法設(shè)計(jì)模式源碼分析。

    4.3.2.3 Bad Code

    錯(cuò)誤的編碼方式:將煮茶的步驟燒水→泡茶→倒水→加檸檬按順序執(zhí)行,煮咖啡的步驟燒水→過(guò)濾咖啡→倒水→加牛奶也按順序執(zhí)行,這樣的缺點(diǎn)是如果步驟很多,那么代碼顯得比較臃腫,代碼維護(hù)成本也會(huì)越來(lái)越高。

    //--------------------------------------代碼塊一--------------------------------------- public class Tea {void prepareRecipe() {boilWater();steepTeaBag();pourInCup();addLemon();}public void boilWater() {System.out.println("Boiling water");}public void steepTeaBag() {System.out.println("Steeping the tea");}public void addLemon() {System.out.println("Adding Lemon");}public void pourInCup() {System.out.println("Pouring into cup");} } //--------------------------------------代碼塊二--------------------------------------- public class Coffee {void prepareRecipe() {boilWater();brewCoffeeGrinds();pourInCup();addSugarAndMilk();}public void boilWater() {System.out.println("Boiling water");}public void brewCoffeeGrinds() {System.out.println("Dripping Coffee through filter");}public void pourInCup() {System.out.println("Pouring into cup");}public void addSugarAndMilk() {System.out.println("Adding Sugar and Milk");} } //--------------------------------------代碼塊三--------------------------------------- public class Barista {public static void main(String[] args) {Tea tea = new Tea();Coffee coffee = new Coffee();System.out.println("Making tea...");tea.prepareRecipe();System.out.println("Making coffee...");coffee.prepareRecipe();} } 復(fù)制代碼

    結(jié)果:

    4.3.2.4 Good Code

    正確的編碼方式:首先將煮茶和煮咖啡共同動(dòng)作燒水和倒水抽取成模板方法,并在父類執(zhí)行,然后煮茶的泡茶、加檸檬步驟,煮咖啡的過(guò)濾咖啡、加牛奶步驟分別差異化實(shí)現(xiàn)即可,最后要確保四個(gè)步驟執(zhí)行鏈準(zhǔn)確性。

    //--------------------------------------代碼塊一--------------------------------------- public abstract class CaffeineBeverage {final void prepareRecipe() {boilWater();brew();pourInCup();addCondiments();}abstract void brew();abstract void addCondiments();void boilWater() {System.out.println("Boiling water");}void pourInCup() {System.out.println("Pouring into cup");} } //--------------------------------------代碼塊二--------------------------------------- public class Tea extends CaffeineBeverage {public void brew() {System.out.println("Steeping the tea");}public void addCondiments() {System.out.println("Adding Lemon");} } //--------------------------------------代碼塊三--------------------------------------- public class Coffee extends CaffeineBeverage {public void brew() {System.out.println("Dripping Coffee through filter");}public void addCondiments() {System.out.println("Adding Sugar and Milk");} } //--------------------------------------代碼塊四--------------------------------------- public class Barista {public static void main(String[] args) {Tea tea = new Tea();Coffee coffee = new Coffee();System.out.println("\nMaking tea...");tea.prepareRecipe();System.out.println("\nMaking coffee...");coffee.prepareRecipe();} 復(fù)制代碼

    結(jié)果:

    4.3.2.5 Source Code

    當(dāng)然Android的AsyncTask也能體現(xiàn)模板方法設(shè)計(jì)模式,我們可以看到execute方法內(nèi)部封裝了onPreExecute, doInBackground, onPostExecute這個(gè)算法框架。

    用戶可以根據(jù)自己的需求來(lái)在覆寫這幾個(gè)方法,使得用戶可以很方便的使用異步任務(wù)來(lái)完成耗時(shí)操作,又可以通過(guò)onPostExecute來(lái)完成更新UI線程的工作。

    //--------------------------------------代碼塊一---------------------------------------public final AsyncTask<Params, Progress, Result> execute(Params... params) {return executeOnExecutor(sDefaultExecutor, params);}public final AsyncTask<Params, Progress, Result> executeOnExecutor(Executor exec,Params... params) { //............................................................................mStatus = Status.RUNNING;// TODO: 關(guān)鍵模板方法onPreExecute();mWorker.mParams = params;exec.execute(mFuture);return this;}//--------------------------------------代碼塊二---------------------------------------public AsyncTask() {mWorker = new WorkerRunnable<Params, Result>() {public Result call() throws Exception {mTaskInvoked.set(true);Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND);// TODO: 關(guān)鍵執(zhí)行方法return postResult(doInBackground(mParams));}};} //--------------------------------------代碼塊三---------------------------------------private Result postResult(Result result) {Message message = sHandler.obtainMessage(MESSAGE_POST_RESULT,new AsyncTaskResult<Result>(this, result));message.sendToTarget();return result;} //--------------------------------------代碼塊四---------------------------------------private static class InternalHandler extends Handler {public void handleMessage(Message msg) {AsyncTaskResult result = (AsyncTaskResult) msg.obj;switch (msg.what) {case MESSAGE_POST_RESULT:result.mTask.finish(result.mData[0]);break;case MESSAGE_POST_PROGRESS:result.mTask.onProgressUpdate(result.mData);break;}}}//--------------------------------------代碼塊五---------------------------------------private void finish(Result result) {if (isCancelled()) {onCancelled(result);} else {// TODO: 關(guān)鍵模板方法onPostExecute(result);}mStatus = Status.FINISHED;} 復(fù)制代碼

    4.3.2.6 注意事項(xiàng)

    當(dāng)然模板方法如果沒(méi)有梳理好方法與方法的調(diào)用鏈關(guān)系,那么模板方法會(huì)帶來(lái)代碼閱讀的難度,會(huì)讓人覺得難以理解。

    五、總結(jié)與展望

    《Android架構(gòu)演進(jìn) · 設(shè)計(jì)模式· 為什么建議你一定要學(xué)透設(shè)計(jì)模式》一文首先通過(guò)5W2H全方位的講解了設(shè)計(jì)模式對(duì)Android開發(fā)的價(jià)值,然后通過(guò)UML圖例、BadCode、Good Code、使用原則和思考復(fù)盤多維度分析了7大設(shè)計(jì)原則優(yōu)劣勢(shì)和核心思想,最后分別對(duì)創(chuàng)建型模式、行為型模式和結(jié)構(gòu)型模式的案例剖析了三大設(shè)計(jì)模式的實(shí)現(xiàn)細(xì)節(jié)。

    因?yàn)槿绻δ芎?jiǎn)單,套用設(shè)計(jì)模式搭建,反而會(huì)增加了成本和系統(tǒng)的復(fù)雜度。因此,在工作中我們既不要生搬硬套設(shè)計(jì)模式,也不要過(guò)度去設(shè)計(jì)。我們要根據(jù)功能需求的復(fù)雜性設(shè)計(jì)系統(tǒng)。

    在理解設(shè)計(jì)模式思想的基礎(chǔ)上,小木箱強(qiáng)烈建議大家結(jié)合框架源碼和項(xiàng)目源碼對(duì)每一個(gè)設(shè)計(jì)模式和設(shè)計(jì)原則,進(jìn)行深度理解和思考,最后才能針對(duì)合適的場(chǎng)景和問(wèn)題正確的運(yùn)用。

    當(dāng)然很多設(shè)計(jì)模式使用場(chǎng)景不是一種模式的唯一實(shí)現(xiàn),可能是多種模式混合實(shí)現(xiàn)。因此,對(duì)Android同學(xué)發(fā)散思維和業(yè)務(wù)理解深度提出苛刻的要求。有的時(shí)候架構(gòu)能力是倒逼的,面對(duì)復(fù)雜的業(yè)務(wù)頻繁的變化,我們要勇于不斷的挑戰(zhàn)!

    這也是小木箱強(qiáng)烈建議大家學(xué)透設(shè)計(jì)模式很重要的原因。希望通過(guò)這篇文章能夠讓你意識(shí)到學(xué)會(huì)設(shè)計(jì)模式的重要性。

    下一章Android架構(gòu)演進(jìn) · 設(shè)計(jì)模式 · Android常見的4種創(chuàng)建型設(shè)計(jì)模式會(huì)從上而下帶大家揭秘常見創(chuàng)建型設(shè)計(jì)模式,我們下一篇見~

    總結(jié)

    以上是生活随笔為你收集整理的Android架构演进 · 设计模式· 为什么建议你一定要学透设计模式?的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。

    美女毛片一区二区三区四区 | 骚片av蜜桃精品一区 | 丰满肥臀大屁股熟妇激情视频 | 国产精品成人av在线观看 | 四虎影视成人永久免费观看视频 | 亚洲成av人综合在线观看 | 在线 国产 欧美 亚洲 天堂 | 亚洲国产精品无码久久久久高潮 | 亚洲一区二区三区偷拍女厕 | 粉嫩少妇内射浓精videos | 欧美日韩人成综合在线播放 | 俄罗斯老熟妇色xxxx | 国产精品久久国产精品99 | 无码午夜成人1000部免费视频 | 熟妇人妻中文av无码 | 精品国偷自产在线 | 中文字幕乱码人妻二区三区 | 久久亚洲日韩精品一区二区三区 | 亚洲娇小与黑人巨大交 | 一本无码人妻在中文字幕免费 | 欧美日韩精品 | 亚洲一区二区三区 | 巨爆乳无码视频在线观看 | 久久精品无码一区二区三区 | 无码国产乱人伦偷精品视频 | 最近中文2019字幕第二页 | 亚洲国产成人av在线观看 | www国产精品内射老师 | 日本在线高清不卡免费播放 | av在线亚洲欧洲日产一区二区 | 无码播放一区二区三区 | 男女超爽视频免费播放 | 一个人看的www免费视频在线观看 | 亚洲午夜无码久久 | 亚洲日韩av一区二区三区中文 | 日日天干夜夜狠狠爱 | 夜夜高潮次次欢爽av女 | www一区二区www免费 | 国产精品鲁鲁鲁 | 欧美大屁股xxxxhd黑色 | 99re在线播放 | 亚洲一区二区三区在线观看网站 | 色婷婷久久一区二区三区麻豆 | 狠狠cao日日穞夜夜穞av | 国产另类ts人妖一区二区 | 国产真实伦对白全集 | 久久精品一区二区三区四区 | 亚洲午夜久久久影院 | 老熟妇仑乱视频一区二区 | 精品夜夜澡人妻无码av蜜桃 | 国产欧美亚洲精品a | 久久国产精品萌白酱免费 | 国产成人精品三级麻豆 | 2020久久超碰国产精品最新 | 亚洲精品综合一区二区三区在线 | 欧美国产日韩亚洲中文 | 台湾无码一区二区 | 扒开双腿吃奶呻吟做受视频 | 国产精品免费大片 | 亚洲日韩精品欧美一区二区 | 午夜无码人妻av大片色欲 | 亚洲日韩一区二区 | 国产精品久免费的黄网站 | 动漫av网站免费观看 | 国产人妻人伦精品1国产丝袜 | 激情内射日本一区二区三区 | 亚洲成色在线综合网站 | 国产情侣作爱视频免费观看 | 色婷婷久久一区二区三区麻豆 | 亚洲阿v天堂在线 | 男女超爽视频免费播放 | 久久国内精品自在自线 | 一区二区三区乱码在线 | 欧洲 | 欧美黑人乱大交 | 成人无码精品一区二区三区 | 午夜理论片yy44880影院 | 99精品无人区乱码1区2区3区 | 色综合久久久无码网中文 | 亚洲一区二区三区香蕉 | 久久无码专区国产精品s | 国产精品.xx视频.xxtv | 精品人人妻人人澡人人爽人人 | 67194成是人免费无码 | 99久久人妻精品免费一区 | 丰满人妻翻云覆雨呻吟视频 | 国产疯狂伦交大片 | 国产在线精品一区二区三区直播 | 丰满少妇女裸体bbw | 精品国产av色一区二区深夜久久 | 国产成人无码av片在线观看不卡 | 人人澡人人透人人爽 | 黑人巨大精品欧美一区二区 | 少妇的肉体aa片免费 | 中文无码成人免费视频在线观看 | 国产成人精品三级麻豆 | 97精品人妻一区二区三区香蕉 | 国产99久久精品一区二区 | 欧美老熟妇乱xxxxx | 国产精品18久久久久久麻辣 | 又湿又紧又大又爽a视频国产 | 十八禁视频网站在线观看 | 无码毛片视频一区二区本码 | 久久精品视频在线看15 | 女人高潮内射99精品 | 中文字幕无码日韩欧毛 | 日本一区二区三区免费高清 | 日日碰狠狠躁久久躁蜜桃 | 日韩无套无码精品 | 国产成人午夜福利在线播放 | 女人高潮内射99精品 | 人人妻人人澡人人爽人人精品 | 国产免费久久久久久无码 | 国产免费久久久久久无码 | 国产精品久久久久久无码 | 3d动漫精品啪啪一区二区中 | 狠狠cao日日穞夜夜穞av | 99精品国产综合久久久久五月天 | 97人妻精品一区二区三区 | 人妻aⅴ无码一区二区三区 | 人妻少妇被猛烈进入中文字幕 | 亚洲自偷精品视频自拍 | 亚洲无人区一区二区三区 | 国产福利视频一区二区 | 色老头在线一区二区三区 | 中文字幕乱码亚洲无线三区 | 搡女人真爽免费视频大全 | 欧美日本精品一区二区三区 | 日日鲁鲁鲁夜夜爽爽狠狠 | 国产一区二区三区日韩精品 | 亚洲一区二区三区四区 | 99国产欧美久久久精品 | 1000部啪啪未满十八勿入下载 | 老子影院午夜精品无码 | 免费无码av一区二区 | 亚洲精品一区国产 | 99久久久国产精品无码免费 | 亚洲日韩av一区二区三区四区 | 亚洲国产精品久久久久久 | 日日摸日日碰夜夜爽av | 18无码粉嫩小泬无套在线观看 | 国产亚洲精品精品国产亚洲综合 | 午夜理论片yy44880影院 | 国产三级精品三级男人的天堂 | 天堂亚洲免费视频 | 国产精品久久久一区二区三区 | 2020久久超碰国产精品最新 | 高清无码午夜福利视频 | 久久久久av无码免费网 | 蜜臀aⅴ国产精品久久久国产老师 | 日韩亚洲欧美精品综合 | 牲欲强的熟妇农村老妇女视频 | 人人妻人人澡人人爽人人精品 | 秋霞成人午夜鲁丝一区二区三区 | 夜夜高潮次次欢爽av女 | 人人妻人人澡人人爽欧美一区九九 | 性生交大片免费看l | 无码人妻出轨黑人中文字幕 | 国产精品国产三级国产专播 | 成人免费视频视频在线观看 免费 | 99久久婷婷国产综合精品青草免费 | 粗大的内捧猛烈进出视频 | 日本熟妇人妻xxxxx人hd | 精品一二三区久久aaa片 | 亚洲综合另类小说色区 | 偷窥村妇洗澡毛毛多 | 午夜无码人妻av大片色欲 | 亚洲娇小与黑人巨大交 | 国产精品久久久久久亚洲影视内衣 | 日本xxxx色视频在线观看免费 | 波多野结衣 黑人 | 国产特级毛片aaaaaa高潮流水 | 国产成人精品一区二区在线小狼 | 蜜臀aⅴ国产精品久久久国产老师 | 人妻互换免费中文字幕 | 日本xxxx色视频在线观看免费 | 日日噜噜噜噜夜夜爽亚洲精品 | 丰满诱人的人妻3 | 18禁黄网站男男禁片免费观看 | 麻豆av传媒蜜桃天美传媒 | 激情人妻另类人妻伦 | 搡女人真爽免费视频大全 | 无码人妻av免费一区二区三区 | а√资源新版在线天堂 | 四虎国产精品一区二区 | 色欲人妻aaaaaaa无码 | 又大又硬又黄的免费视频 | 国产福利视频一区二区 | 欧洲熟妇精品视频 | 久久www免费人成人片 | 国产午夜手机精彩视频 | 少妇性荡欲午夜性开放视频剧场 | 乱人伦人妻中文字幕无码久久网 | 图片区 小说区 区 亚洲五月 | 精品国精品国产自在久国产87 | 欧美亚洲日韩国产人成在线播放 | 熟女少妇人妻中文字幕 | 国产亚洲精品久久久久久久 | 国内揄拍国内精品人妻 | 精品成人av一区二区三区 | 一本久道久久综合狠狠爱 | 人人妻人人澡人人爽人人精品浪潮 | 国产成人午夜福利在线播放 | 妺妺窝人体色www在线小说 | 国产精品亚洲专区无码不卡 | 桃花色综合影院 | 3d动漫精品啪啪一区二区中 | 国产成人无码专区 | 久久久久亚洲精品男人的天堂 | 国产激情一区二区三区 | 国产午夜手机精彩视频 | 狠狠色噜噜狠狠狠狠7777米奇 | 少妇激情av一区二区 | 伊人久久大香线蕉av一区二区 | 成人性做爰aaa片免费看 | 免费中文字幕日韩欧美 | 久久久久久亚洲精品a片成人 | 成人一区二区免费视频 | 久久久亚洲欧洲日产国码αv | 双乳奶水饱满少妇呻吟 | av香港经典三级级 在线 | 亚洲午夜久久久影院 | 色婷婷欧美在线播放内射 | 性生交大片免费看女人按摩摩 | 高潮毛片无遮挡高清免费视频 | 狠狠色噜噜狠狠狠7777奇米 | 大肉大捧一进一出好爽视频 | 欧美高清在线精品一区 | 国内精品久久毛片一区二区 | 无码免费一区二区三区 | 国产成人一区二区三区在线观看 | 欧美性黑人极品hd | 日本www一道久久久免费榴莲 | 99久久人妻精品免费二区 | 久久综合九色综合97网 | 久久精品中文字幕大胸 | 又大又紧又粉嫩18p少妇 | 少女韩国电视剧在线观看完整 | 九九久久精品国产免费看小说 | 麻花豆传媒剧国产免费mv在线 | 免费无码肉片在线观看 | 中文字幕无码视频专区 | 在线观看国产一区二区三区 | 国产农村妇女aaaaa视频 撕开奶罩揉吮奶头视频 | 久久熟妇人妻午夜寂寞影院 | 亚洲色www成人永久网址 | 亚洲国产av美女网站 | 久久久www成人免费毛片 | 国产精品igao视频网 | 国产乱人无码伦av在线a | 97无码免费人妻超级碰碰夜夜 | 国产av人人夜夜澡人人爽麻豆 | 六月丁香婷婷色狠狠久久 | 中文字幕无码视频专区 | 中文字幕人妻无码一夲道 | 久久综合九色综合97网 | 激情五月综合色婷婷一区二区 | 欧美日本日韩 | 久久精品中文字幕一区 | 久久人人爽人人人人片 | 中文字幕精品av一区二区五区 | 欧美三级不卡在线观看 | 人妻少妇精品视频专区 | 99久久精品无码一区二区毛片 | 东京热男人av天堂 | 爆乳一区二区三区无码 | 亚洲人成无码网www | 波多野结衣高清一区二区三区 | 国产成人综合色在线观看网站 | 久久精品99久久香蕉国产色戒 | 三级4级全黄60分钟 | 免费播放一区二区三区 | 亚洲中文字幕乱码av波多ji | 久久伊人色av天堂九九小黄鸭 | 精品午夜福利在线观看 | 国产精品久久久一区二区三区 | 欧美丰满熟妇xxxx | 久久亚洲中文字幕无码 | 欧美一区二区三区视频在线观看 | 久久99国产综合精品 | 国产99久久精品一区二区 | 人妻少妇被猛烈进入中文字幕 | 国产在线一区二区三区四区五区 | 久久精品中文闷骚内射 | 色综合久久久无码中文字幕 | 综合激情五月综合激情五月激情1 | 波多野结衣一区二区三区av免费 | 丝袜美腿亚洲一区二区 | 中文字幕乱码人妻二区三区 | 免费观看激色视频网站 | 麻豆国产人妻欲求不满谁演的 | 国产三级精品三级男人的天堂 | 精品厕所偷拍各类美女tp嘘嘘 | 丰满人妻一区二区三区免费视频 | 亚洲欧洲无卡二区视頻 | 人人爽人人爽人人片av亚洲 | 帮老师解开蕾丝奶罩吸乳网站 | 国产精品亚洲综合色区韩国 | 永久免费观看国产裸体美女 | аⅴ资源天堂资源库在线 | 欧美野外疯狂做受xxxx高潮 | 女人被爽到呻吟gif动态图视看 | 精品国产一区二区三区av 性色 | 精品人妻人人做人人爽夜夜爽 | 久久久中文久久久无码 | 久久99精品国产麻豆 | 中国女人内谢69xxxxxa片 | 国产三级精品三级男人的天堂 | 在线观看免费人成视频 | 婷婷丁香五月天综合东京热 | 特大黑人娇小亚洲女 | 精品无码av一区二区三区 | 蜜桃视频插满18在线观看 | 日韩人妻系列无码专区 | 久久精品成人欧美大片 | 亚洲熟女一区二区三区 | 麻豆精产国品 | 国产精品香蕉在线观看 | 日本www一道久久久免费榴莲 | 中文毛片无遮挡高清免费 | 人妻中文无码久热丝袜 | 成人三级无码视频在线观看 | 天天做天天爱天天爽综合网 | 久久久久se色偷偷亚洲精品av | 亚洲精品一区国产 | 国产精品igao视频网 | 香港三级日本三级妇三级 | 国产午夜无码视频在线观看 | 强开小婷嫩苞又嫩又紧视频 | 久久无码专区国产精品s | 国产精品亚洲一区二区三区喷水 | 人妻少妇精品无码专区动漫 | 亚洲七七久久桃花影院 | 中文字幕 亚洲精品 第1页 | 亚洲中文字幕成人无码 | 国产又粗又硬又大爽黄老大爷视 | 久久久av男人的天堂 | 成人亚洲精品久久久久软件 | 精品国产一区二区三区av 性色 | 99久久无码一区人妻 | 欧美性生交活xxxxxdddd | 日韩精品一区二区av在线 | 亚洲成熟女人毛毛耸耸多 | 久久久久久国产精品无码下载 | 久久成人a毛片免费观看网站 | 人人妻人人澡人人爽人人精品浪潮 | 中文字幕av无码一区二区三区电影 | 久久综合香蕉国产蜜臀av | 2020久久超碰国产精品最新 | 精品国产精品久久一区免费式 | 国产乡下妇女做爰 | 亚洲精品午夜国产va久久成人 | 久久国产精品二国产精品 | 内射巨臀欧美在线视频 | 日韩人妻无码一区二区三区久久99 | 日本熟妇乱子伦xxxx | 欧美日韩一区二区三区自拍 | 国产精品美女久久久久av爽李琼 | 丰满少妇女裸体bbw | 亚洲人成无码网www | 日日摸天天摸爽爽狠狠97 | 亚洲国产精品久久久久久 | 99久久久国产精品无码免费 | 无码人妻出轨黑人中文字幕 | 在线视频网站www色 | 国产成人精品一区二区在线小狼 | 爽爽影院免费观看 | 女高中生第一次破苞av | 疯狂三人交性欧美 | 99久久人妻精品免费二区 | 国产精品成人av在线观看 | 日本熟妇大屁股人妻 | 免费人成网站视频在线观看 | 精品乱子伦一区二区三区 | 国产亚洲tv在线观看 | 色欲人妻aaaaaaa无码 | 丰满人妻翻云覆雨呻吟视频 | 国产凸凹视频一区二区 | 久久久国产一区二区三区 | 99精品视频在线观看免费 | 久久人人爽人人人人片 | 在教室伦流澡到高潮hnp视频 | 国产无套粉嫩白浆在线 | 日本欧美一区二区三区乱码 | 国产精品久久久午夜夜伦鲁鲁 | 精品无人区无码乱码毛片国产 | 亚洲中文字幕在线无码一区二区 | 亚洲日本va午夜在线电影 | 免费中文字幕日韩欧美 | 欧美三级a做爰在线观看 | 成人毛片一区二区 | 国产免费观看黄av片 | 欧美自拍另类欧美综合图片区 | 一本久久a久久精品亚洲 | 日日天日日夜日日摸 | 草草网站影院白丝内射 | 夜夜高潮次次欢爽av女 | 俄罗斯老熟妇色xxxx | 久久久久成人片免费观看蜜芽 | 理论片87福利理论电影 | 中文字幕无码视频专区 | 午夜嘿嘿嘿影院 | 狂野欧美性猛xxxx乱大交 | 色一情一乱一伦 | 国产无遮挡又黄又爽免费视频 | 人人澡人摸人人添 | 狂野欧美性猛xxxx乱大交 | 99久久精品日本一区二区免费 | 亚洲中文字幕在线观看 | 俄罗斯老熟妇色xxxx | 色一情一乱一伦一区二区三欧美 | 在教室伦流澡到高潮hnp视频 | ass日本丰满熟妇pics | 国产精品久久久久久无码 | 亚洲国产精品无码久久久久高潮 | av无码久久久久不卡免费网站 | 一本久久伊人热热精品中文字幕 | 伊人久久大香线蕉亚洲 | 中文字幕+乱码+中文字幕一区 | 久青草影院在线观看国产 | 国产成人综合在线女婷五月99播放 | 国产又爽又黄又刺激的视频 | 夜先锋av资源网站 | 国产av无码专区亚洲a∨毛片 | 狠狠色噜噜狠狠狠狠7777米奇 | 我要看www免费看插插视频 | 亚洲爆乳精品无码一区二区三区 | 日日天日日夜日日摸 | 国产一区二区不卡老阿姨 | 久久午夜夜伦鲁鲁片无码免费 | 国产免费久久精品国产传媒 | yw尤物av无码国产在线观看 | 欧美色就是色 | 99久久亚洲精品无码毛片 | 国产精品第一区揄拍无码 | 国产sm调教视频在线观看 | 影音先锋中文字幕无码 | 999久久久国产精品消防器材 | 色 综合 欧美 亚洲 国产 | 欧美日韩人成综合在线播放 | 欧美熟妇另类久久久久久多毛 | 成人免费视频一区二区 | 自拍偷自拍亚洲精品10p | 波多野42部无码喷潮在线 | 偷窥村妇洗澡毛毛多 | 久久久精品欧美一区二区免费 | 色诱久久久久综合网ywww | 人人妻人人藻人人爽欧美一区 | 国产亚洲tv在线观看 | 成人aaa片一区国产精品 | 我要看www免费看插插视频 | 国产乱人无码伦av在线a | 女人被男人躁得好爽免费视频 | 色诱久久久久综合网ywww | 久久综合网欧美色妞网 | 国产人妻久久精品二区三区老狼 | 久久www免费人成人片 | 99久久亚洲精品无码毛片 | 国产精品美女久久久久av爽李琼 | 欧美老熟妇乱xxxxx | 亚洲国产精品久久久天堂 | 国产欧美精品一区二区三区 | 欧美 日韩 人妻 高清 中文 | 久久人人爽人人爽人人片av高清 | 国产人妻大战黑人第1集 | 国产午夜无码视频在线观看 | 成人精品一区二区三区中文字幕 | 丰满人妻翻云覆雨呻吟视频 | 国产内射老熟女aaaa | 51国偷自产一区二区三区 | 日日碰狠狠躁久久躁蜜桃 | 精品国产精品久久一区免费式 | 亚洲色大成网站www国产 | 成人免费视频视频在线观看 免费 | 99久久精品日本一区二区免费 | 国产 精品 自在自线 | 亚洲欧美色中文字幕在线 | 亚洲伊人久久精品影院 | 97无码免费人妻超级碰碰夜夜 | 欧美日韩人成综合在线播放 | 内射欧美老妇wbb | 免费人成在线观看网站 | 熟妇人妻激情偷爽文 | 乱人伦中文视频在线观看 | 少妇的肉体aa片免费 | 婷婷丁香五月天综合东京热 | 99久久无码一区人妻 | 欧美一区二区三区视频在线观看 | 乱人伦人妻中文字幕无码久久网 | 亚洲精品国产第一综合99久久 | 亚洲欧美中文字幕5发布 | 极品尤物被啪到呻吟喷水 | 人妻无码久久精品人妻 | 乌克兰少妇性做爰 | 丰满少妇熟乱xxxxx视频 | 中文字幕无码av激情不卡 | 精品水蜜桃久久久久久久 | 国产乡下妇女做爰 | 欧美性猛交内射兽交老熟妇 | 又粗又大又硬又长又爽 | 欧美激情综合亚洲一二区 | 日日天干夜夜狠狠爱 | 国产成人综合色在线观看网站 | √天堂中文官网8在线 | 99久久婷婷国产综合精品青草免费 | 一本久道久久综合婷婷五月 | 国产精品第一国产精品 | 亚洲啪av永久无码精品放毛片 | 亚洲狠狠婷婷综合久久 | 国内丰满熟女出轨videos | 成熟女人特级毛片www免费 | 久久综合激激的五月天 | 成人欧美一区二区三区黑人免费 | 中文字幕无线码免费人妻 | 老熟妇乱子伦牲交视频 | 老头边吃奶边弄进去呻吟 | 久久综合色之久久综合 | 色窝窝无码一区二区三区色欲 | 性欧美大战久久久久久久 | 性生交大片免费看女人按摩摩 | 久久精品视频在线看15 | 18禁黄网站男男禁片免费观看 | 国产人妻精品一区二区三区 | 97久久国产亚洲精品超碰热 | 宝宝好涨水快流出来免费视频 | 国产香蕉97碰碰久久人人 | 国产又爽又猛又粗的视频a片 | 日韩精品一区二区av在线 | av无码不卡在线观看免费 | 久久久久99精品国产片 | 日本护士毛茸茸高潮 | 少妇人妻大乳在线视频 | 免费观看激色视频网站 | 国产免费久久精品国产传媒 | 国内精品人妻无码久久久影院 | 国产9 9在线 | 中文 | 熟妇人妻无码xxx视频 | 日本一区二区更新不卡 | 人妻互换免费中文字幕 | 国产手机在线αⅴ片无码观看 | 无码毛片视频一区二区本码 | 狠狠噜狠狠狠狠丁香五月 | 日日橹狠狠爱欧美视频 | 国产成人精品必看 | 性色欲网站人妻丰满中文久久不卡 | 一二三四社区在线中文视频 | 国产成人一区二区三区别 | 免费观看又污又黄的网站 | 天天躁日日躁狠狠躁免费麻豆 | 国产性生大片免费观看性 | 国产午夜福利亚洲第一 | 性欧美疯狂xxxxbbbb | 在线a亚洲视频播放在线观看 | 水蜜桃亚洲一二三四在线 | 99久久婷婷国产综合精品青草免费 | 一本色道婷婷久久欧美 | 国产办公室秘书无码精品99 | 国产精品亚洲专区无码不卡 | 久久国语露脸国产精品电影 | 女人高潮内射99精品 | 亚洲中文字幕在线观看 | 国产精品内射视频免费 | 免费观看的无遮挡av | 麻豆人妻少妇精品无码专区 | 精品aⅴ一区二区三区 | 性欧美熟妇videofreesex | 国产成人一区二区三区在线观看 | 人妻aⅴ无码一区二区三区 | 国产suv精品一区二区五 | 色五月丁香五月综合五月 | 欧美熟妇另类久久久久久不卡 | 波多野结衣乳巨码无在线观看 | 国产欧美熟妇另类久久久 | 免费人成网站视频在线观看 | 亚洲成av人片天堂网无码】 | 国产精品人人爽人人做我的可爱 | 老司机亚洲精品影院 | 无遮挡国产高潮视频免费观看 | 亚洲欧美综合区丁香五月小说 | 无码吃奶揉捏奶头高潮视频 | 久久精品国产99久久6动漫 | 免费人成在线观看网站 | 欧美人妻一区二区三区 | 少女韩国电视剧在线观看完整 | 成人精品视频一区二区三区尤物 | 国产精品第一国产精品 | 在线观看欧美一区二区三区 | 无码人妻久久一区二区三区不卡 | 国产艳妇av在线观看果冻传媒 | 久久综合给合久久狠狠狠97色 | 真人与拘做受免费视频一 | 少妇太爽了在线观看 | 婷婷丁香六月激情综合啪 | 久久久www成人免费毛片 | 国产黄在线观看免费观看不卡 | 伊人久久大香线蕉亚洲 | 精品人妻人人做人人爽夜夜爽 | 九九在线中文字幕无码 | 欧美三级不卡在线观看 | 人人爽人人澡人人人妻 | 亚洲成a人片在线观看无码3d | 欧美日本精品一区二区三区 | 麻豆精品国产精华精华液好用吗 | 又粗又大又硬又长又爽 | 内射欧美老妇wbb | 日本熟妇人妻xxxxx人hd | 少妇太爽了在线观看 | a片免费视频在线观看 | 中文字幕无码免费久久99 | 丝袜 中出 制服 人妻 美腿 | 欧美乱妇无乱码大黄a片 | 美女黄网站人色视频免费国产 | 国产成人无码午夜视频在线观看 | 思思久久99热只有频精品66 | 国产猛烈高潮尖叫视频免费 | 人妻少妇被猛烈进入中文字幕 | av人摸人人人澡人人超碰下载 | 久久综合网欧美色妞网 | 日本精品高清一区二区 | 人妻与老人中文字幕 | 亚洲精品国产精品乱码不卡 | 午夜性刺激在线视频免费 | 亚洲小说图区综合在线 | 久久 国产 尿 小便 嘘嘘 | 性史性农村dvd毛片 | 无套内谢老熟女 | 色窝窝无码一区二区三区色欲 | 国产婷婷色一区二区三区在线 | 亚洲精品一区三区三区在线观看 | 麻豆蜜桃av蜜臀av色欲av | 国精产品一品二品国精品69xx | 领导边摸边吃奶边做爽在线观看 | 日韩少妇内射免费播放 | 自拍偷自拍亚洲精品被多人伦好爽 | 欧美成人高清在线播放 | 女人被男人躁得好爽免费视频 | 日本一卡二卡不卡视频查询 | 国精产品一品二品国精品69xx | 国产麻豆精品一区二区三区v视界 | 人妻互换免费中文字幕 | 欧美激情一区二区三区成人 | 日本免费一区二区三区最新 | 日本一本二本三区免费 | 午夜福利不卡在线视频 | 性开放的女人aaa片 | 秋霞成人午夜鲁丝一区二区三区 | 成人免费无码大片a毛片 | 亚洲の无码国产の无码步美 | 色欲久久久天天天综合网精品 | 大乳丰满人妻中文字幕日本 | 色 综合 欧美 亚洲 国产 | 女人被男人爽到呻吟的视频 | 亚洲热妇无码av在线播放 | 欧美野外疯狂做受xxxx高潮 | 1000部夫妻午夜免费 | 无码成人精品区在线观看 | 国产精品内射视频免费 | 成人av无码一区二区三区 | 熟妇人妻中文av无码 | 日本在线高清不卡免费播放 | 亚洲а∨天堂久久精品2021 | 久久综合香蕉国产蜜臀av | 中文字幕精品av一区二区五区 | 精品欧洲av无码一区二区三区 | 亚洲色在线无码国产精品不卡 | 日本一区二区三区免费播放 | 欧美人与善在线com | 欧美丰满熟妇xxxx | 熟女俱乐部五十路六十路av | 欧美zoozzooz性欧美 | 中文字幕+乱码+中文字幕一区 | 色综合久久中文娱乐网 | 国产精品久久久 | 又大又硬又黄的免费视频 | 人妻有码中文字幕在线 | 欧美国产日韩久久mv | 国精产品一品二品国精品69xx | 欧美自拍另类欧美综合图片区 | 日本乱人伦片中文三区 | 天堂а√在线地址中文在线 | 蜜桃臀无码内射一区二区三区 | 久久熟妇人妻午夜寂寞影院 | 国产精品18久久久久久麻辣 | 天天爽夜夜爽夜夜爽 | 大肉大捧一进一出好爽视频 | 久久久久av无码免费网 | 国产精华av午夜在线观看 | 纯爱无遮挡h肉动漫在线播放 | 熟妇人妻中文av无码 | 色综合久久中文娱乐网 | 久久综合给合久久狠狠狠97色 | 国产乱人无码伦av在线a | 宝宝好涨水快流出来免费视频 | 日韩精品无码免费一区二区三区 | 牲欲强的熟妇农村老妇女 | 精品熟女少妇av免费观看 | 久久 国产 尿 小便 嘘嘘 | 精品夜夜澡人妻无码av蜜桃 | 99精品无人区乱码1区2区3区 | 西西人体www44rt大胆高清 | 亚洲а∨天堂久久精品2021 | 久激情内射婷内射蜜桃人妖 | 国产又爽又猛又粗的视频a片 | 亚洲国精产品一二二线 | 色综合久久中文娱乐网 | 帮老师解开蕾丝奶罩吸乳网站 | 精品 日韩 国产 欧美 视频 | 少妇人妻偷人精品无码视频 | 在线精品国产一区二区三区 | 国产情侣作爱视频免费观看 | 麻豆成人精品国产免费 | √天堂资源地址中文在线 | 在线视频网站www色 | 99久久久无码国产aaa精品 | 精品国产精品久久一区免费式 | 丁香啪啪综合成人亚洲 | 无码av岛国片在线播放 | 成人欧美一区二区三区黑人免费 | 天堂а√在线地址中文在线 | 久久精品国产大片免费观看 | 人人妻人人澡人人爽欧美一区 | 国产午夜亚洲精品不卡 | 乱人伦人妻中文字幕无码久久网 | 色一情一乱一伦 | 国产精品爱久久久久久久 | av在线亚洲欧洲日产一区二区 | 日本饥渴人妻欲求不满 | 丰满少妇高潮惨叫视频 | 国产成人久久精品流白浆 | 欧美日韩一区二区免费视频 | 亚洲无人区一区二区三区 | 中文亚洲成a人片在线观看 | 国产女主播喷水视频在线观看 | 俺去俺来也www色官网 | 18禁止看的免费污网站 | 久久久亚洲欧洲日产国码αv | 日韩少妇内射免费播放 | 2019nv天堂香蕉在线观看 | 麻豆果冻传媒2021精品传媒一区下载 | 日韩人妻少妇一区二区三区 | 国产精品无码一区二区三区不卡 | 人妻aⅴ无码一区二区三区 | 国产精品久久久久9999小说 | 奇米综合四色77777久久 东京无码熟妇人妻av在线网址 | 亚洲第一网站男人都懂 | 亚洲一区二区三区在线观看网站 | 久久久久久久久蜜桃 | 好屌草这里只有精品 | 日本丰满护士爆乳xxxx | 精品成人av一区二区三区 | 九月婷婷人人澡人人添人人爽 | 清纯唯美经典一区二区 | 欧美日本日韩 | 无码人妻精品一区二区三区不卡 | 亚洲爆乳无码专区 | 国产一区二区不卡老阿姨 | 黑人巨大精品欧美一区二区 | 国产又爽又猛又粗的视频a片 | 欧美精品国产综合久久 | 久久久婷婷五月亚洲97号色 | 亚洲一区av无码专区在线观看 | 欧洲精品码一区二区三区免费看 | 色综合久久中文娱乐网 | 午夜熟女插插xx免费视频 | 午夜性刺激在线视频免费 | 狂野欧美激情性xxxx | 国产精品久久福利网站 | 国内精品久久久久久中文字幕 | 人妻尝试又大又粗久久 | 精品成人av一区二区三区 | 天海翼激烈高潮到腰振不止 | 亚洲精品午夜无码电影网 | 亚欧洲精品在线视频免费观看 | 在线а√天堂中文官网 | 精品厕所偷拍各类美女tp嘘嘘 | 国产午夜福利100集发布 | 欧美自拍另类欧美综合图片区 | 大肉大捧一进一出好爽视频 | 亚洲国产成人av在线观看 | 国产香蕉尹人视频在线 | 97资源共享在线视频 | 成人精品一区二区三区中文字幕 | 欧美三级不卡在线观看 | 18禁止看的免费污网站 | 人人妻人人藻人人爽欧美一区 | 老熟女重囗味hdxx69 | 亚洲第一无码av无码专区 | 无码乱肉视频免费大全合集 | 亚洲无人区午夜福利码高清完整版 | 欧美熟妇另类久久久久久不卡 | 少妇性俱乐部纵欲狂欢电影 | 久久99精品久久久久久动态图 | 国产婷婷色一区二区三区在线 | 国产av一区二区三区最新精品 | 国产欧美熟妇另类久久久 | 欧美阿v高清资源不卡在线播放 | 亚洲一区二区三区香蕉 | 婷婷综合久久中文字幕蜜桃三电影 | 99精品久久毛片a片 | 日本一区二区三区免费播放 | 久久亚洲精品中文字幕无男同 | 国内少妇偷人精品视频 | 久久久中文字幕日本无吗 | 国产精品.xx视频.xxtv | 国产人妻大战黑人第1集 | 狠狠综合久久久久综合网 | 狠狠色色综合网站 | 性欧美牲交在线视频 | 少妇人妻大乳在线视频 | 中文字幕无码热在线视频 | 免费无码午夜福利片69 | 日韩精品久久久肉伦网站 | 久久精品人人做人人综合 | 国产无遮挡又黄又爽免费视频 | 日本一卡二卡不卡视频查询 | 老熟妇乱子伦牲交视频 | 18禁止看的免费污网站 | 日本一区二区三区免费高清 | 人妻少妇被猛烈进入中文字幕 | 激情爆乳一区二区三区 | 国产无遮挡吃胸膜奶免费看 | 国产精品人妻一区二区三区四 | 中文字幕 亚洲精品 第1页 | 久久精品国产精品国产精品污 | 国产人妻精品一区二区三区 | 无码精品国产va在线观看dvd | 国产办公室秘书无码精品99 | 内射白嫩少妇超碰 | 国产福利视频一区二区 | 国内精品久久久久久中文字幕 | 国产乱人偷精品人妻a片 | 成人性做爰aaa片免费看不忠 | 国产精品人人爽人人做我的可爱 | 国产在线无码精品电影网 | 久久综合九色综合97网 | 久久五月精品中文字幕 | 强伦人妻一区二区三区视频18 | 国产97色在线 | 免 | 欧美性生交活xxxxxdddd | 5858s亚洲色大成网站www | 在线精品亚洲一区二区 | 亚洲精品美女久久久久久久 | 久久综合给久久狠狠97色 | 精品一区二区三区波多野结衣 | 亚洲色欲久久久综合网东京热 | 国产成人一区二区三区在线观看 | 夜夜躁日日躁狠狠久久av | 1000部啪啪未满十八勿入下载 | 精品一区二区三区波多野结衣 | 久久99国产综合精品 | 亚洲va中文字幕无码久久不卡 | 国内综合精品午夜久久资源 | 久久国产劲爆∧v内射 | 奇米影视7777久久精品 | 2020久久香蕉国产线看观看 | 国产性生大片免费观看性 | 18黄暴禁片在线观看 | 无遮无挡爽爽免费视频 | 性色欲网站人妻丰满中文久久不卡 | 国产高清不卡无码视频 | 国产无av码在线观看 | 正在播放东北夫妻内射 | 亚洲天堂2017无码中文 | 亚洲国产综合无码一区 | 无码纯肉视频在线观看 | 图片小说视频一区二区 | 四虎国产精品一区二区 | 国产性猛交╳xxx乱大交 国产精品久久久久久无码 欧洲欧美人成视频在线 | 国产无遮挡又黄又爽又色 | 日韩精品a片一区二区三区妖精 | 久久综合九色综合97网 | 日日橹狠狠爱欧美视频 | 国产片av国语在线观看 | 欧美 日韩 人妻 高清 中文 | 亚洲小说图区综合在线 | 大地资源网第二页免费观看 | 少妇的肉体aa片免费 | 亚洲色在线无码国产精品不卡 | 日本精品久久久久中文字幕 | 在线观看国产一区二区三区 | 欧美日韩久久久精品a片 | 日韩av无码一区二区三区不卡 | 亚洲成a人片在线观看无码3d | 久久综合九色综合欧美狠狠 | 免费网站看v片在线18禁无码 | 精品国产国产综合精品 | 精品人妻人人做人人爽夜夜爽 | 色综合久久中文娱乐网 | 国产免费久久久久久无码 | 日日夜夜撸啊撸 | 日本精品少妇一区二区三区 | 国产精品对白交换视频 | 国产精品爱久久久久久久 | 国产精品免费大片 | 美女扒开屁股让男人桶 | 97资源共享在线视频 | 麻豆av传媒蜜桃天美传媒 | 国产人妻人伦精品1国产丝袜 | 亚洲国产日韩a在线播放 | 久久综合激激的五月天 | 国产精品99久久精品爆乳 | 久久国产精品精品国产色婷婷 | 免费无码一区二区三区蜜桃大 | 天堂久久天堂av色综合 | 99精品国产综合久久久久五月天 | 夫妻免费无码v看片 | 狂野欧美性猛xxxx乱大交 | 一本色道久久综合狠狠躁 | 久久99精品国产.久久久久 | 骚片av蜜桃精品一区 | 日本爽爽爽爽爽爽在线观看免 | 国产乱人伦偷精品视频 | 成人女人看片免费视频放人 | 18黄暴禁片在线观看 | 极品嫩模高潮叫床 | 中文字幕无码免费久久99 | 中文字幕人妻无码一区二区三区 | 国产特级毛片aaaaaaa高清 | 日本高清一区免费中文视频 | 日本大香伊一区二区三区 | 日本大乳高潮视频在线观看 | 久久亚洲中文字幕精品一区 | 国产乱人伦av在线无码 | 亚洲日本一区二区三区在线 | 丰满少妇熟乱xxxxx视频 | 三级4级全黄60分钟 | 国产激情艳情在线看视频 | 国模大胆一区二区三区 | 国产做国产爱免费视频 | 高清无码午夜福利视频 | 内射爽无广熟女亚洲 | 亚洲成a人片在线观看无码 | 无码一区二区三区在线观看 | 最近的中文字幕在线看视频 | 亚洲成av人在线观看网址 | 久精品国产欧美亚洲色aⅴ大片 | 国产偷抇久久精品a片69 | 成人性做爰aaa片免费看 | 精品国产乱码久久久久乱码 | 无码av中文字幕免费放 | 人妻少妇被猛烈进入中文字幕 | 中文亚洲成a人片在线观看 | 日本熟妇乱子伦xxxx | 欧美成人高清在线播放 | 日本乱人伦片中文三区 | aa片在线观看视频在线播放 | 亚洲一区二区三区含羞草 | 国产成人综合在线女婷五月99播放 | 福利一区二区三区视频在线观看 | 成人aaa片一区国产精品 | 女人高潮内射99精品 | 无码精品国产va在线观看dvd | 无码av中文字幕免费放 | 久久综合九色综合97网 | 欧美丰满熟妇xxxx | 麻豆精品国产精华精华液好用吗 | 成在人线av无码免费 | 熟妇人妻无乱码中文字幕 | 人妻无码αv中文字幕久久琪琪布 | 亚洲熟悉妇女xxx妇女av | 黄网在线观看免费网站 | 亚洲 欧美 激情 小说 另类 | 国产精品第一国产精品 | 六十路熟妇乱子伦 | 色一情一乱一伦一区二区三欧美 | 97精品人妻一区二区三区香蕉 | 国产精品理论片在线观看 | 天堂无码人妻精品一区二区三区 | 一本久道久久综合狠狠爱 | 无码人妻丰满熟妇区五十路百度 | 久久国产精品二国产精品 | 一本色道久久综合狠狠躁 | 在线a亚洲视频播放在线观看 | 国产精品亚洲综合色区韩国 | 日韩欧美中文字幕在线三区 | 天天拍夜夜添久久精品大 | 亚洲啪av永久无码精品放毛片 | 夜精品a片一区二区三区无码白浆 | 强辱丰满人妻hd中文字幕 | 亚洲区欧美区综合区自拍区 | 午夜男女很黄的视频 | 精品无码成人片一区二区98 | 亚洲成a人片在线观看日本 | 色窝窝无码一区二区三区色欲 | 性欧美大战久久久久久久 | 久久人人爽人人爽人人片ⅴ | 女人和拘做爰正片视频 | 中文字幕无线码免费人妻 | 亚洲精品久久久久久一区二区 | 99国产欧美久久久精品 | 欧美日韩人成综合在线播放 | 日韩少妇白浆无码系列 | 国产高清av在线播放 | 波多野结衣一区二区三区av免费 | 亚洲爆乳精品无码一区二区三区 | 色综合久久久久综合一本到桃花网 | 99在线 | 亚洲 | 丰满少妇熟乱xxxxx视频 | 国产成人人人97超碰超爽8 | 久久国产精品偷任你爽任你 | 97精品人妻一区二区三区香蕉 | 四虎影视成人永久免费观看视频 | 澳门永久av免费网站 | 巨爆乳无码视频在线观看 | 俄罗斯老熟妇色xxxx | 日韩无码专区 | 日韩人妻无码中文字幕视频 | 国产成人精品必看 | 免费网站看v片在线18禁无码 | 精品久久8x国产免费观看 | 久久国产精品精品国产色婷婷 | 丰满妇女强制高潮18xxxx | 国产激情一区二区三区 | 无码纯肉视频在线观看 | 国产美女极度色诱视频www | 日本精品高清一区二区 | 日本一本二本三区免费 | 国产午夜视频在线观看 | 永久免费观看美女裸体的网站 | 国产精品丝袜黑色高跟鞋 | 午夜无码区在线观看 | 无码人妻丰满熟妇区五十路百度 | 日本大乳高潮视频在线观看 | 77777熟女视频在线观看 а天堂中文在线官网 | 欧美日本免费一区二区三区 | 无人区乱码一区二区三区 | 国产69精品久久久久app下载 | 久久国产36精品色熟妇 | 成人精品一区二区三区中文字幕 | 免费无码一区二区三区蜜桃大 | 久久久久久亚洲精品a片成人 | 久久国产精品偷任你爽任你 | 欧洲熟妇精品视频 | 久久精品国产99精品亚洲 | 色欲综合久久中文字幕网 | 欧洲欧美人成视频在线 | 成人三级无码视频在线观看 | 国产9 9在线 | 中文 | √天堂中文官网8在线 | 日日天干夜夜狠狠爱 | 亚洲爆乳大丰满无码专区 | 丰满护士巨好爽好大乳 | 久久zyz资源站无码中文动漫 | 性色欲网站人妻丰满中文久久不卡 | 毛片内射-百度 | 强开小婷嫩苞又嫩又紧视频 | 亚洲色欲色欲天天天www | 乌克兰少妇性做爰 | 97夜夜澡人人双人人人喊 | 熟妇人妻激情偷爽文 | 国产成人亚洲综合无码 | 国产手机在线αⅴ片无码观看 | 中文字幕无码免费久久9一区9 | 日本www一道久久久免费榴莲 | 人人超人人超碰超国产 | 色欲人妻aaaaaaa无码 | 婷婷色婷婷开心五月四房播播 | 国产区女主播在线观看 | 亚洲一区二区观看播放 | 久久久久久av无码免费看大片 | 日本熟妇人妻xxxxx人hd | 免费播放一区二区三区 | 午夜精品久久久内射近拍高清 | 亚洲 日韩 欧美 成人 在线观看 | 欧美日韩久久久精品a片 | 亚洲国产欧美日韩精品一区二区三区 | 人妻尝试又大又粗久久 | 人人妻在人人 | 无码人妻少妇伦在线电影 | 网友自拍区视频精品 | 成人性做爰aaa片免费看 | 亚洲爆乳精品无码一区二区三区 | 精品成人av一区二区三区 | 又色又爽又黄的美女裸体网站 | 久久无码中文字幕免费影院蜜桃 | 夜夜高潮次次欢爽av女 | a片在线免费观看 | 人人妻人人藻人人爽欧美一区 | 日日天日日夜日日摸 | 亚洲第一网站男人都懂 | 国产无遮挡又黄又爽免费视频 | 中文字幕无码av激情不卡 | 人妻插b视频一区二区三区 | 奇米影视7777久久精品 | 日本精品久久久久中文字幕 | 亚洲 另类 在线 欧美 制服 | 99久久久无码国产精品免费 | 欧美自拍另类欧美综合图片区 | 成人亚洲精品久久久久软件 | 美女黄网站人色视频免费国产 | 高潮毛片无遮挡高清免费 | 天天av天天av天天透 | aⅴ亚洲 日韩 色 图网站 播放 | 国产性猛交╳xxx乱大交 国产精品久久久久久无码 欧洲欧美人成视频在线 | 日日躁夜夜躁狠狠躁 | 亚洲精品欧美二区三区中文字幕 | 少妇无码一区二区二三区 | 久久综合九色综合欧美狠狠 | 狠狠综合久久久久综合网 | 国产艳妇av在线观看果冻传媒 | 亚洲熟女一区二区三区 | 国产精品嫩草久久久久 | 大肉大捧一进一出好爽视频 | 好男人www社区 | 东北女人啪啪对白 | 久久午夜夜伦鲁鲁片无码免费 | 国产电影无码午夜在线播放 | 99久久久国产精品无码免费 | 久久www免费人成人片 | 亚洲熟熟妇xxxx | 久久精品国产99精品亚洲 | 人妻夜夜爽天天爽三区 | 俺去俺来也在线www色官网 | 成熟人妻av无码专区 | 国产av无码专区亚洲awww | 男人的天堂2018无码 | 无码av岛国片在线播放 | 亚洲综合久久一区二区 | 日韩视频 中文字幕 视频一区 | 精品国产青草久久久久福利 | 亚洲成色在线综合网站 | 7777奇米四色成人眼影 | 欧美黑人性暴力猛交喷水 | 久久亚洲精品成人无码 | 国产精品久久精品三级 | 午夜时刻免费入口 | 国产亚洲精品久久久闺蜜 | 中国女人内谢69xxxx | 国产av一区二区三区最新精品 | 亚洲无人区午夜福利码高清完整版 | 国产精品香蕉在线观看 | 伊人久久大香线蕉亚洲 | 九月婷婷人人澡人人添人人爽 | 国产精品久久久 | 国产人妻精品一区二区三区不卡 | 欧美精品无码一区二区三区 | 人妻少妇精品视频专区 | 无码吃奶揉捏奶头高潮视频 | aa片在线观看视频在线播放 | 夜夜躁日日躁狠狠久久av | 亚洲 高清 成人 动漫 | 欧美亚洲日韩国产人成在线播放 | 正在播放东北夫妻内射 | 青草青草久热国产精品 | 一二三四在线观看免费视频 | 国产97人人超碰caoprom | 免费观看激色视频网站 | 亚洲精品一区三区三区在线观看 | 伊人久久大香线蕉亚洲 | 日韩精品无码免费一区二区三区 | 亚洲成av人片天堂网无码】 | 久久精品女人的天堂av | 国产va免费精品观看 | √8天堂资源地址中文在线 | 18禁止看的免费污网站 | 日日摸日日碰夜夜爽av | 国产成人精品必看 | 久久久www成人免费毛片 | 中文字幕av伊人av无码av | 人妻少妇精品无码专区二区 | √天堂中文官网8在线 | 无码精品人妻一区二区三区av | 亚洲精品国偷拍自产在线观看蜜桃 | 国产亚洲精品久久久ai换 | 未满小14洗澡无码视频网站 | 强伦人妻一区二区三区视频18 | 亚洲а∨天堂久久精品2021 | 国内精品一区二区三区不卡 | 久久久久se色偷偷亚洲精品av | 黑人大群体交免费视频 | 麻豆国产人妻欲求不满 | 成人精品一区二区三区中文字幕 | 人妻少妇精品视频专区 | 波多野结衣aⅴ在线 | 熟女俱乐部五十路六十路av | 国产人妻精品午夜福利免费 | 日本熟妇大屁股人妻 | 国产色xx群视频射精 | aa片在线观看视频在线播放 | 成人性做爰aaa片免费看 | 国产av一区二区精品久久凹凸 | 欧美日韩人成综合在线播放 | 1000部啪啪未满十八勿入下载 | 精品日本一区二区三区在线观看 | av在线亚洲欧洲日产一区二区 | 精品国产青草久久久久福利 | 帮老师解开蕾丝奶罩吸乳网站 | 国产做国产爱免费视频 | 国产猛烈高潮尖叫视频免费 | 日本一卡二卡不卡视频查询 | 黑人粗大猛烈进出高潮视频 | 亚洲va欧美va天堂v国产综合 | 精品国产精品久久一区免费式 | 漂亮人妻洗澡被公强 日日躁 | 成人无码精品1区2区3区免费看 | 欧美丰满老熟妇xxxxx性 | 色综合久久久无码中文字幕 | 国产av剧情md精品麻豆 | 1000部夫妻午夜免费 | 欧美人妻一区二区三区 | 天堂亚洲2017在线观看 | 国产精品va在线播放 | 波多野结衣乳巨码无在线观看 | 国产精品无套呻吟在线 | 福利一区二区三区视频在线观看 | 亚洲午夜无码久久 | 麻豆人妻少妇精品无码专区 | 国产精品高潮呻吟av久久 | 精品久久久无码人妻字幂 | 欧美亚洲日韩国产人成在线播放 | 久久久中文久久久无码 | 国产超碰人人爽人人做人人添 | 日本肉体xxxx裸交 | 欧美成人午夜精品久久久 | 久久久中文字幕日本无吗 | 亚洲精品美女久久久久久久 | 亚洲欧美日韩成人高清在线一区 | 国产成人无码av片在线观看不卡 | 精品无码成人片一区二区98 | 国产人成高清在线视频99最全资源 | 亚洲a无码综合a国产av中文 | 成年美女黄网站色大免费视频 | 色一情一乱一伦一区二区三欧美 | 夜夜影院未满十八勿进 | 亚洲中文字幕在线无码一区二区 | 性欧美大战久久久久久久 | 一本久久a久久精品亚洲 | 荫蒂被男人添的好舒服爽免费视频 | 无码毛片视频一区二区本码 | 免费无码一区二区三区蜜桃大 | 国产乱人伦av在线无码 | 国产9 9在线 | 中文 | 亚洲区小说区激情区图片区 | 午夜精品一区二区三区的区别 | 美女张开腿让人桶 | 午夜丰满少妇性开放视频 | 国产一精品一av一免费 | 任你躁在线精品免费 | 国产成人一区二区三区在线观看 | 亚洲人成网站在线播放942 | 国语自产偷拍精品视频偷 | 亚洲国产一区二区三区在线观看 | 国产成人精品必看 | аⅴ资源天堂资源库在线 | 亚洲中文无码av永久不收费 | 国产精品a成v人在线播放 | 欧美色就是色 | 国产成人无码专区 | 四虎国产精品免费久久 | 国产成人久久精品流白浆 | 日日麻批免费40分钟无码 | 亚洲色欲色欲欲www在线 | 网友自拍区视频精品 | 久久国内精品自在自线 | 老司机亚洲精品影院无码 | 国产97在线 | 亚洲 | 亚洲国产精品一区二区美利坚 | 欧美国产亚洲日韩在线二区 | 亚洲a无码综合a国产av中文 | 婷婷六月久久综合丁香 | 精品亚洲成av人在线观看 | 少妇性荡欲午夜性开放视频剧场 | 国产乱人无码伦av在线a | 免费无码肉片在线观看 | 精品久久综合1区2区3区激情 | 暴力强奷在线播放无码 | 爱做久久久久久 | 一本久道高清无码视频 | 国产精品办公室沙发 | 久久伊人色av天堂九九小黄鸭 | 51国偷自产一区二区三区 | 精品乱子伦一区二区三区 | 少妇性俱乐部纵欲狂欢电影 | 中文字幕av日韩精品一区二区 | 国产凸凹视频一区二区 | 色妞www精品免费视频 | 精品偷自拍另类在线观看 | 一本色道久久综合狠狠躁 | 国产成人精品视频ⅴa片软件竹菊 | 国产精品久久久 | 黑人巨大精品欧美黑寡妇 | 无码成人精品区在线观看 | 欧美激情一区二区三区成人 | 欧洲熟妇色 欧美 | 国产成人精品三级麻豆 | 九九久久精品国产免费看小说 | 97资源共享在线视频 | 亚洲一区二区三区播放 | 国产情侣作爱视频免费观看 | 图片区 小说区 区 亚洲五月 | 人妻天天爽夜夜爽一区二区 | 综合人妻久久一区二区精品 | 欧美亚洲国产一区二区三区 | √天堂中文官网8在线 | 99久久精品日本一区二区免费 | 少妇人妻av毛片在线看 | 国产另类ts人妖一区二区 | 香蕉久久久久久av成人 | 国产乱人伦偷精品视频 | 日本va欧美va欧美va精品 | 性做久久久久久久免费看 | 人妻插b视频一区二区三区 | 国产精品国产自线拍免费软件 | 色综合久久久无码网中文 | 无套内谢的新婚少妇国语播放 | 久久久成人毛片无码 | 欧美自拍另类欧美综合图片区 | 欧美人与动性行为视频 | 色五月丁香五月综合五月 | 亚洲一区二区三区播放 | 欧美刺激性大交 | 亚洲熟熟妇xxxx | 中文亚洲成a人片在线观看 | 成人免费视频在线观看 | 水蜜桃色314在线观看 | 国产热a欧美热a在线视频 | 亚洲欧美日韩综合久久久 | 无码人妻出轨黑人中文字幕 | 国产乱人无码伦av在线a | 麻豆精品国产精华精华液好用吗 | 欧美日韩综合一区二区三区 | 国产真人无遮挡作爱免费视频 | 国产精品沙发午睡系列 | 国产精品久久久午夜夜伦鲁鲁 | 日欧一片内射va在线影院 | 人人妻人人澡人人爽人人精品 | 欧美喷潮久久久xxxxx | 日韩少妇内射免费播放 | 人妻少妇精品视频专区 | 日韩欧美成人免费观看 | 特级做a爰片毛片免费69 | 国产日产欧产精品精品app | 欧美乱妇无乱码大黄a片 | 曰韩无码二三区中文字幕 | 鲁一鲁av2019在线 | 少妇一晚三次一区二区三区 | 国产无遮挡吃胸膜奶免费看 | 精品偷自拍另类在线观看 | 国产精品久久久一区二区三区 | 一本加勒比波多野结衣 | 免费人成网站视频在线观看 | 蜜臀av在线观看 在线欧美精品一区二区三区 | 又色又爽又黄的美女裸体网站 | 女人被爽到呻吟gif动态图视看 | 天堂久久天堂av色综合 | 国产卡一卡二卡三 | 97精品国产97久久久久久免费 | 久久97精品久久久久久久不卡 | 欧美zoozzooz性欧美 | 日韩成人一区二区三区在线观看 | 国产莉萝无码av在线播放 | 无码福利日韩神码福利片 | 在线欧美精品一区二区三区 | 精品国偷自产在线 | 亚洲最大成人网站 | 丰满少妇高潮惨叫视频 | 国内丰满熟女出轨videos | 色情久久久av熟女人妻网站 | 特黄特色大片免费播放器图片 | 国产午夜视频在线观看 | 天天拍夜夜添久久精品大 | 精品国偷自产在线视频 | 亚洲自偷自拍另类第1页 | 国产三级久久久精品麻豆三级 | 精品人妻中文字幕有码在线 | 成年美女黄网站色大免费视频 | 无码国模国产在线观看 | 亚洲日韩av一区二区三区四区 | 真人与拘做受免费视频一 | 国内综合精品午夜久久资源 | 骚片av蜜桃精品一区 | 亚洲成a人片在线观看日本 | 香港三级日本三级妇三级 | 久久午夜夜伦鲁鲁片无码免费 | 日本xxxx色视频在线观看免费 | 成人欧美一区二区三区 | 天堂一区人妻无码 | 影音先锋中文字幕无码 | 亚洲 欧美 激情 小说 另类 | 亚洲人成网站色7799 | 日本丰满熟妇videos | 国产色在线 | 国产 | 久久精品国产一区二区三区 | 97无码免费人妻超级碰碰夜夜 | 欧美人与禽zoz0性伦交 | 久久 国产 尿 小便 嘘嘘 | 88国产精品欧美一区二区三区 | 男女猛烈xx00免费视频试看 | 大地资源中文第3页 | 国产乱子伦视频在线播放 | 欧美 亚洲 国产 另类 | 色综合视频一区二区三区 | 六月丁香婷婷色狠狠久久 | 精品国产一区av天美传媒 | 波多野结衣一区二区三区av免费 | 性生交大片免费看女人按摩摩 | 亚洲精品久久久久中文第一幕 | 亚洲综合在线一区二区三区 | 国产麻豆精品精东影业av网站 | 亚洲人成网站在线播放942 | 亚洲色在线无码国产精品不卡 | 国产成人久久精品流白浆 | 国产真实伦对白全集 | 3d动漫精品啪啪一区二区中 | 亚洲热妇无码av在线播放 | 宝宝好涨水快流出来免费视频 | 欧美日本免费一区二区三区 | 中文字幕无码人妻少妇免费 | 国产欧美熟妇另类久久久 | 中国女人内谢69xxxx | 日本精品人妻无码免费大全 | 一本色道久久综合亚洲精品不卡 | 日韩无套无码精品 | 激情爆乳一区二区三区 | 国产午夜福利100集发布 | 学生妹亚洲一区二区 | 亚洲人成人无码网www国产 | 国产亚洲欧美日韩亚洲中文色 | 无码国模国产在线观看 | 久激情内射婷内射蜜桃人妖 | 国产精品资源一区二区 | 88国产精品欧美一区二区三区 | 久久久婷婷五月亚洲97号色 | 亚洲国产精品久久久天堂 | 在线播放亚洲第一字幕 | 亚洲人成网站色7799 | 一本久久a久久精品亚洲 | 国产熟妇另类久久久久 | 精品午夜福利在线观看 | 樱花草在线社区www | 日本精品少妇一区二区三区 | 亚洲乱码国产乱码精品精 | 爽爽影院免费观看 | 国产av剧情md精品麻豆 | av人摸人人人澡人人超碰下载 | 综合网日日天干夜夜久久 | 久久久久国色av免费观看性色 | 精品日本一区二区三区在线观看 | 欧洲熟妇色 欧美 | 久久精品中文字幕大胸 | 性色av无码免费一区二区三区 | 中文字幕无线码 | 久久久久久亚洲精品a片成人 | 在线亚洲高清揄拍自拍一品区 | 亚洲欧洲日本综合aⅴ在线 | 亚洲 另类 在线 欧美 制服 | 日日摸夜夜摸狠狠摸婷婷 | 台湾无码一区二区 | 亚洲成a人一区二区三区 | 青草青草久热国产精品 | 99久久99久久免费精品蜜桃 | 中文字幕av日韩精品一区二区 | 无码乱肉视频免费大全合集 | 白嫩日本少妇做爰 | 欧美野外疯狂做受xxxx高潮 | 精品国产青草久久久久福利 | 色婷婷综合激情综在线播放 | 亚洲 日韩 欧美 成人 在线观看 | 国产精品成人av在线观看 | 久久99精品国产麻豆 | 中文字幕av日韩精品一区二区 | 中文无码伦av中文字幕 | 日产国产精品亚洲系列 | 综合网日日天干夜夜久久 | аⅴ资源天堂资源库在线 | 成人三级无码视频在线观看 | 一本大道久久东京热无码av | 又湿又紧又大又爽a视频国产 | 野外少妇愉情中文字幕 | 国产乱人偷精品人妻a片 | 亚洲热妇无码av在线播放 | 久久亚洲精品成人无码 | 免费看男女做好爽好硬视频 | 亚洲日本va午夜在线电影 | 久久亚洲中文字幕无码 | 亚洲精品一区二区三区在线 | 77777熟女视频在线观看 а天堂中文在线官网 | 日本欧美一区二区三区乱码 | 初尝人妻少妇中文字幕 | 国产成人无码区免费内射一片色欲 | 国产精品18久久久久久麻辣 | 国产美女精品一区二区三区 | 国产69精品久久久久app下载 | 亚洲成av人综合在线观看 | 国产成人一区二区三区在线观看 | 久久精品女人天堂av免费观看 | 免费观看黄网站 | 欧美 亚洲 国产 另类 | 一个人免费观看的www视频 | 精品久久8x国产免费观看 | 1000部夫妻午夜免费 | 丰满少妇女裸体bbw | 国产超碰人人爽人人做人人添 | 天天做天天爱天天爽综合网 | 西西人体www44rt大胆高清 | 色窝窝无码一区二区三区色欲 | 欧美性黑人极品hd | 久久精品国产一区二区三区 | 中文字幕av日韩精品一区二区 | 色综合久久中文娱乐网 | 鲁一鲁av2019在线 | 亚洲欧美国产精品专区久久 | 久久精品中文字幕大胸 | 4hu四虎永久在线观看 | 亚洲日韩中文字幕在线播放 | 99精品无人区乱码1区2区3区 | 亚洲欧美综合区丁香五月小说 | 男人和女人高潮免费网站 | 国产疯狂伦交大片 | 无码人妻丰满熟妇区五十路百度 | 99久久久无码国产精品免费 | 久久久国产精品无码免费专区 | 嫩b人妻精品一区二区三区 | 久久 国产 尿 小便 嘘嘘 | 国产亚洲视频中文字幕97精品 | 内射白嫩少妇超碰 | 一本久久a久久精品vr综合 | 好爽又高潮了毛片免费下载 | 亚洲精品综合一区二区三区在线 | 99精品无人区乱码1区2区3区 | 国产亚洲人成在线播放 | 国产在线aaa片一区二区99 | 国内精品一区二区三区不卡 | 一区二区传媒有限公司 | 精品亚洲韩国一区二区三区 | 亚洲中文字幕无码中文字在线 | 国産精品久久久久久久 | 国产成人无码a区在线观看视频app | 又湿又紧又大又爽a视频国产 | 欧洲vodafone精品性 | 一本久道久久综合狠狠爱 | 搡女人真爽免费视频大全 | 强奷人妻日本中文字幕 | 亚洲国产一区二区三区在线观看 | 中文字幕日产无线码一区 | 亚洲另类伦春色综合小说 | 高潮毛片无遮挡高清免费 | 亚洲熟妇自偷自拍另类 | 亚洲男人av香蕉爽爽爽爽 | 天堂一区人妻无码 | 亚洲国产精品毛片av不卡在线 | 亚洲成在人网站无码天堂 | 久久国产精品_国产精品 | 久久久久亚洲精品男人的天堂 | 一个人看的www免费视频在线观看 | 中文字幕无码乱人伦 | 99久久无码一区人妻 | 国产精品爱久久久久久久 | yw尤物av无码国产在线观看 | 亚洲一区二区三区在线观看网站 | 欧美激情内射喷水高潮 | 东京热一精品无码av | 亚洲精品一区二区三区在线观看 | 久久久精品456亚洲影院 | 国产精华av午夜在线观看 | 人妻少妇被猛烈进入中文字幕 | 国内精品久久久久久中文字幕 | 3d动漫精品啪啪一区二区中 | 性欧美疯狂xxxxbbbb | 久久国产精品_国产精品 | 亚洲国产成人a精品不卡在线 | 国产熟女一区二区三区四区五区 | 亚洲精品成人av在线 | 成人一在线视频日韩国产 | 亚洲国产欧美日韩精品一区二区三区 | 欧美乱妇无乱码大黄a片 | 性啪啪chinese东北女人 | 曰韩无码二三区中文字幕 | 无码人妻精品一区二区三区不卡 | 精品久久久久久人妻无码中文字幕 | 大乳丰满人妻中文字幕日本 | 亚洲s色大片在线观看 | 精品久久8x国产免费观看 | 日韩人妻系列无码专区 | 国产精品高潮呻吟av久久4虎 | 在线观看国产一区二区三区 | 亚洲中文字幕久久无码 | 曰韩无码二三区中文字幕 | 久久久久99精品国产片 | 国产亚av手机在线观看 | 色综合久久久久综合一本到桃花网 | 色欲综合久久中文字幕网 | 男人的天堂av网站 | 无码国产激情在线观看 | 成人无码精品一区二区三区 | 纯爱无遮挡h肉动漫在线播放 | 无码午夜成人1000部免费视频 | 精品亚洲韩国一区二区三区 | 人人超人人超碰超国产 | 内射老妇bbwx0c0ck | 97夜夜澡人人双人人人喊 | 日韩精品一区二区av在线 | 97人妻精品一区二区三区 | 沈阳熟女露脸对白视频 | 亚洲色欲久久久综合网东京热 | 无码福利日韩神码福利片 | 国产9 9在线 | 中文 | 大地资源网第二页免费观看 | 奇米综合四色77777久久 东京无码熟妇人妻av在线网址 | 精品夜夜澡人妻无码av蜜桃 | 在线a亚洲视频播放在线观看 | 国产无遮挡吃胸膜奶免费看 | 亚洲无人区午夜福利码高清完整版 | 亚洲日韩一区二区三区 | 两性色午夜视频免费播放 | 丰腴饱满的极品熟妇 | 中文字幕亚洲情99在线 | 国产午夜亚洲精品不卡 | 日韩精品a片一区二区三区妖精 | 国产熟妇另类久久久久 | 精品 日韩 国产 欧美 视频 | 精品国产青草久久久久福利 | 97久久超碰中文字幕 | 呦交小u女精品视频 | 国产精品亚洲一区二区三区喷水 | 丰满少妇人妻久久久久久 | 亚洲乱码国产乱码精品精 | 99久久99久久免费精品蜜桃 | 夫妻免费无码v看片 | 久久久久久久女国产乱让韩 | 欧美日韩久久久精品a片 | 国产内射老熟女aaaa | 欧美成人免费全部网站 | 国产超级va在线观看视频 | 亚洲综合精品香蕉久久网 | 在线a亚洲视频播放在线观看 | 成人无码视频在线观看网站 | 亚洲成在人网站无码天堂 | 精品少妇爆乳无码av无码专区 | 一个人看的www免费视频在线观看 | 无套内谢的新婚少妇国语播放 | 青青草原综合久久大伊人精品 | 人妻少妇精品无码专区动漫 | 人妻插b视频一区二区三区 | 亚洲人成网站色7799 | 国产亚洲视频中文字幕97精品 | 亚洲 日韩 欧美 成人 在线观看 | 国产suv精品一区二区五 | 久久综合久久自在自线精品自 | 综合人妻久久一区二区精品 | 曰韩无码二三区中文字幕 | 久久久久久av无码免费看大片 | 久久成人a毛片免费观看网站 | 理论片87福利理论电影 | 夫妻免费无码v看片 | 日本丰满熟妇videos | 少妇一晚三次一区二区三区 | 中文字幕av伊人av无码av | 国产精品成人av在线观看 | 亚洲成a人片在线观看无码3d | 国内综合精品午夜久久资源 | 青青青手机频在线观看 | 在线播放无码字幕亚洲 | 97久久国产亚洲精品超碰热 | 亚洲自偷自偷在线制服 | 色诱久久久久综合网ywww | 一本无码人妻在中文字幕免费 | 任你躁在线精品免费 | 久久99热只有频精品8 | 丝袜足控一区二区三区 | 亚洲熟妇自偷自拍另类 | 色欲av亚洲一区无码少妇 | 欧美老人巨大xxxx做受 | 2020最新国产自产精品 | 人人超人人超碰超国产 | 中文字幕乱码亚洲无线三区 | 麻豆人妻少妇精品无码专区 | 粉嫩少妇内射浓精videos | 久久久精品成人免费观看 | 六十路熟妇乱子伦 | 亚洲日韩一区二区 | 国产精品亚洲а∨无码播放麻豆 | 国产精品久久久久7777 | 人人妻人人澡人人爽欧美一区 | 日日天干夜夜狠狠爱 | 欧美 日韩 亚洲 在线 | 无人区乱码一区二区三区 |