冰浪

          哥已不再年輕 - 堅(jiān)定夢(mèng)想,畢生追求!
          posts - 85, comments - 90, trackbacks - 0, articles - 3
            BlogJava :: 首頁(yè) :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理

          在我看來,Template Method(模板方法) 和 Factory Method (工廠方法)有許多相似的地方。

          兩種模式共同點(diǎn)在于:
          1.類定義:首先都是先定義一個(gè)接口(父類),并定義其抽象方法。然后建立子類繼承父類,并實(shí)現(xiàn)父類所定義的方法。
          2.使用:以接口(父類)為變量類型,使用子類生成對(duì)象實(shí)例,然后此對(duì)象實(shí)例調(diào)用接口的方法(抽象方法,子類已實(shí)現(xiàn))進(jìn)行工作。

          不同點(diǎn)在于:
          Template Method 生成對(duì)象實(shí)例是直接生成的(new),即是用子類的構(gòu)造函數(shù)生成。而 Factory Method? 的“實(shí)體”對(duì)象實(shí)例并不是直接new出來的,而只能通過工廠(Factory類)生成。就像工廠生產(chǎn)產(chǎn)品一樣,產(chǎn)品本身并不會(huì)憑空自己出來,而只能是工廠造出來。而且Factory Method? 比Template Method 多了個(gè)Factory 類。

          所以,在 Factory Method? 中,子類的構(gòu)造函數(shù)都是定義成 private,而并不定義成 public 。

          從這里讓我想到了Java中接口和繼承的實(shí)質(zhì)。接口和繼承的實(shí)質(zhì)并不在于簡(jiǎn)單的方法繼承和代碼重用,真正的目的是抽象!

          在這里,我轉(zhuǎn)貼一篇文章,更清楚地認(rèn)識(shí)接口的實(shí)質(zhì)。我想很多人都讀過吧。


          //----注:以下文章為轉(zhuǎn)貼----

          澄清Java語(yǔ)言接口與繼承的本質(zhì)

          大多數(shù)人認(rèn)為,接口的意義在于頂替多重繼承。眾所周知Java沒有c++那樣多重繼承的機(jī)制,但是卻能夠?qū)嵶鞫鄠€(gè)接口。其實(shí)這樣做是很牽強(qiáng)的,接口和繼承是完全不同的東西,接口沒有能力代替多重繼承,也沒有這個(gè)義務(wù)。接口的作用,一言以蔽之,就是標(biāo)志類的類別(type of class)。把不同類型的類歸于不同的接口,可以更好的管理他們。OO的精髓,我以為,是對(duì)對(duì)象的抽象,最能體現(xiàn)這一點(diǎn)的就是接口。為什么我們討論設(shè)計(jì)模式都只針對(duì)具備了抽象能力的語(yǔ)言(比如c++、java、c#等),就是因?yàn)樵O(shè)計(jì)模式所研究的,實(shí)際上就是如何合理的去抽象。(cowboy的名言是“抽象就是抽去像的部分”,看似調(diào)侃,實(shí)乃至理)。

            設(shè)計(jì)模式中最基礎(chǔ)的是工廠模式(Factory),在我最近的一個(gè)很簡(jiǎn)單的應(yīng)用中,我想盡量的讓我的程序能夠在多個(gè)數(shù)據(jù)庫(kù)間移植,當(dāng)然,這涉及很多問題,單是如何兼容不同DBMS的SQL就讓人頭痛。我們不妨先把問題簡(jiǎn)單化,只考慮如何連接不同的數(shù)據(jù)庫(kù)。

            假設(shè)我有很多個(gè)類,分別是Mysql.java、SQLServer.java、Oracle.java、DB2.java,他們分別連接不同的數(shù)據(jù)庫(kù),統(tǒng)一返回一個(gè)Connection對(duì)象,并且都有一個(gè)close方法,用于關(guān)閉連接。只需要針對(duì)你的DBMS,選擇不同的類,就可以用了,但是我的用戶他會(huì)使用什么數(shù)據(jù)庫(kù)?我不知道,我希望的是盡量少的修改代碼,就能滿足他的需要。我可以抽象如下接口:

          package org.bromon.test;
          public interface DB
          {
            java.sql.Connection openDB(String url,String user,String password);
            void close();
          }

            這個(gè)接口只定義兩個(gè)方法,沒有任何有實(shí)際意義的代碼,具體的代碼由實(shí)作這個(gè)接口的類來給出,比如Mysql.java:

          Package org.bromon.test;
          import java.sql.*;
          public class Mysql implements DB
          {
            private String url=”jdbc:mysql:localhost:3306/test”;
            private String user=”root”;
            private String password=””;
            private Connection conn;
            public Connection openDB(url,user,password)
            {
              //連接數(shù)據(jù)庫(kù)的代碼
            }

            public void close()
            {
              //關(guān)閉數(shù)據(jù)庫(kù)
            }
          }

            類似的當(dāng)然還有Oracle.java等等,接口DB給這些類歸了個(gè)類,在應(yīng)用程序中我們這樣定義對(duì)象:

            org.bromon.test.DB myDB;

            使用myDB來操作數(shù)據(jù)庫(kù),就可以不用管實(shí)際上我所使用的是哪個(gè)類,這就是所謂的“開-閉”原則。但是問題在于接口是不能實(shí)例化的,myDB=new DB(),這樣的代碼是絕對(duì)錯(cuò)誤的,我們只能myDB=new Mysql()或者myDB=new Oracle()。麻煩了,我還是需要指定具體實(shí)例化的是哪個(gè)類,用了接口跟沒用一樣。所以我們需要一個(gè)工廠:

          package org.bromon.test;
          public class DBFactory
          {
            public static DB Connection getConn()
            {
              Return(new Mysql());
            }
          }

            所以實(shí)例化的代碼變成:myDB=DBFactory.getConn();

            這就是23種模式中最基礎(chǔ)的普通工廠(Factory),工廠類負(fù)責(zé)具體實(shí)例化哪個(gè)類,而其他的程序邏輯都是針對(duì)DB這個(gè)接口進(jìn)行操作,這就是“針對(duì)接口編程”。責(zé)任都被推卸給工廠類了,當(dāng)然你也可以繼續(xù)定義工廠接口,繼續(xù)把責(zé)任上拋,這就演變成抽象工廠(Abstract Factory)。

            整個(gè)過程中接口不負(fù)責(zé)任何具體操作,其他的程序要連接數(shù)據(jù)庫(kù)的話,只需要構(gòu)造一個(gè)DB對(duì)象就OK,而不管工廠類如何變化。這就是接口的意義----抽象。

            繼承的概念不用多說,很好理解。為什么要繼承呢?因?yàn)槟阆胫赜么a?這絕對(duì)不是理由,繼承的意義也在于抽象,而不是代碼重用。如果對(duì)象A有一個(gè)run()方法,對(duì)象B也想有這個(gè)方法,所以有人就Class B extends A。這是不經(jīng)大腦的做法。如果在B中實(shí)例化一個(gè)A,調(diào)用A的Run()方法,是不是可以達(dá)到同樣的目的?如下:

          Class B
          {
            A a=new A();
            a.run();
          }

            這就是利用類的聚合來重用代碼,是委派模式的雛形,是GoF一貫倡導(dǎo)的做法。

            那么繼承的意義何在?其實(shí)這是歷史原因造成的,最開始的OO語(yǔ)言只有繼承,沒有接口,所以只能以繼承來實(shí)現(xiàn)抽象,請(qǐng)一定注意,繼承的本意在于抽象,而非代碼重用(雖然繼承也有這個(gè)作用),這是很多Java爛書最嚴(yán)重的錯(cuò)誤之一,它們所造成的陰影,我至今還沒有完全擺脫,壞書害人啊,尤其是入門類的,流毒太大。什么時(shí)候應(yīng)該使用繼承?只在抽象類中使用,其他情況下盡量不使用。抽象類也是不能實(shí)例化的,它僅僅提供一個(gè)模版而已,這就很能說明問題。

            軟件開發(fā)的萬(wàn)惡之源,一是重復(fù)代碼而不是重用代碼,二是爛用繼承,尤以c++程序員為甚。Java中取締多重繼承,目的就是制止?fàn)€用繼承,實(shí)是非常明智的做法,不過很多人都不理解。Java能夠更好的體現(xiàn)設(shè)計(jì),這是讓我入迷的原因之一.


          評(píng)論

          # re: 設(shè)計(jì)模式學(xué)習(xí)筆記之三:Template Method 和 Factory Method  回復(fù)  更多評(píng)論   

          2006-09-08 21:53 by Alex
          " 設(shè)計(jì)模式中最基礎(chǔ)的是工廠模式(Factory)" 有道理。
          "繼承的意義也在于抽象"是的,OO有一條原則就是聚合優(yōu)于繼承是同樣道理

          另外,關(guān)于模板方式,實(shí)際就是在抽象類中留部分需要在子類中實(shí)現(xiàn)的,父類只提供一個(gè)基本的流程,即method()的組合,能實(shí)現(xiàn)的就先實(shí)現(xiàn),不能實(shí)現(xiàn)的就留給子類.

          # re: 設(shè)計(jì)模式學(xué)習(xí)筆記之三:Template Method 和 Factory Method  回復(fù)  更多評(píng)論   

          2006-09-13 21:09 by 區(qū)別很大的
          factory 是創(chuàng)建性模式 template 是行為模式。template method 也不一定要使用 factory method 。 特別是一個(gè) template method 類不涉及其他類的時(shí)候。

          # re: 設(shè)計(jì)模式學(xué)習(xí)筆記之三:Template Method 和 Factory Method  回復(fù)  更多評(píng)論   

          2006-09-14 12:57 by 冰浪
          呵呵,本人只是初探模式之人,所以想法并不一定正確。
          我這里說的“相似”是指兩種模式的具體實(shí)現(xiàn)的細(xì)節(jié)上,并不是指運(yùn)用。還望各位多多指點(diǎn)!

          # re: 設(shè)計(jì)模式學(xué)習(xí)筆記之三:Template Method 和 Factory Method  回復(fù)  更多評(píng)論   

          2006-09-15 14:09 by lijiajia418
          "OO有一條原則就是聚合優(yōu)于繼承是同樣道理"
          所有從基類繼承的子類型都擁有父類所定義的行為,如果當(dāng)子類型不具備某一抽象父類中定義的行為時(shí),那么這時(shí)我們就應(yīng)該優(yōu)先考慮組合,而不是在子類型中繼續(xù)繼承父類的行為,擴(kuò)展子類的行為,造成接口的污染,這種設(shè)計(jì)是嚴(yán)重違反OO的。
          主站蜘蛛池模板: 隆回县| 巩留县| 海晏县| 东安县| 重庆市| 井冈山市| 通州区| 郸城县| 天峻县| 塘沽区| 宝清县| 古田县| 惠州市| 乃东县| 西乡县| 石柱| 和硕县| 青海省| 蒙自县| 通山县| 高邮市| 云林县| 双流县| 秦皇岛市| 永州市| 彭泽县| 泾川县| 肥东县| 盐源县| 交口县| 麻江县| 宁城县| 固始县| 麦盖提县| 绿春县| 页游| 冀州市| 巨鹿县| 林口县| 南京市| 瑞昌市|