<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    posts - 89,  comments - 98,  trackbacks - 0

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

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

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

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

      這個接口只定義兩個方法,沒有任何有實(shí)際意義的代碼,具體的代碼由實(shí)作這個接口的類來給出,比如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ù)庫的代碼
      }

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

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

      org.bromon.test.DB myDB;

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

    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í)例化哪個類,而其他的程序邏輯都是針對DB這個接口進(jìn)行操作,這就是“針對接口編程”。責(zé)任都被推卸給工廠類了,當(dāng)然你也可以繼續(xù)定義工廠接口,繼續(xù)把責(zé)任上拋,這就演變成抽象工廠(Abstract Factory)。

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

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

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

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

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

      軟件開發(fā)的萬惡之源,一是重復(fù)代碼而不是重用代碼,二是爛用繼承,尤以c++程序員為甚。Java中取締多重繼承,目的就是制止?fàn)€用繼承,實(shí)是非常明智的做法,不過很多人都不理解。Java能夠更好的體現(xiàn)設(shè)計(jì),這是讓我入迷的原因之一。
    posted on 2006-10-26 08:55 水煮三國 閱讀(577) 評論(3)  編輯  收藏 所屬分類: J2SE

    FeedBack:
    # re: 澄清Java語言接口與繼承的本質(zhì)
    2006-10-26 09:28 | potomo
    受益非淺。  回復(fù)  更多評論
      
    # re: 澄清Java語言接口與繼承的本質(zhì)
    2006-10-26 17:43 | cjren[匿名]
    問個問題:在
    “public class DBFactory
    {
      public static DB Connection getConn()
      {
        Return(new Mysql());
      }
    }

      所以實(shí)例化的代碼變成:myDB=DBFactory.getConn();”
    中我得到的是Mysql的實(shí)例啊?怎么才得到Oracle的實(shí)例呢?也就是說工廠類中怎么判斷該返回哪個合適的實(shí)例對象呢?
    謝謝  回復(fù)  更多評論
      
    # re: 澄清Java語言接口與繼承的本質(zhì)
    2006-10-26 17:50 | cjren[匿名]
    的確在我看到的書中都是提倡能用聚合就不用繼承。
    我還想問,
    1,接口和抽象類的關(guān)鍵區(qū)別點(diǎn)在哪里呢?接口只是為了更純粹的抽象嗎?
    2,怎么才能算是好的體現(xiàn)了代碼重用呢?Java語言中哪些特性的目的是代碼重用的呢?package算否否呢?
    謝謝啊:-)  回復(fù)  更多評論
      
    <2006年10月>
    24252627282930
    1234567
    891011121314
    15161718192021
    22232425262728
    2930311234

    常用鏈接

    留言簿(4)

    隨筆分類(85)

    隨筆檔案(89)

    文章分類(14)

    文章檔案(42)

    收藏夾(37)

    java

    oracle

    Sybase

    搜索

    •  

    積分與排名

    • 積分 - 210785
    • 排名 - 266

    最新評論

    閱讀排行榜

    評論排行榜

    主站蜘蛛池模板: 亚洲一区二区在线免费观看| 性做久久久久久免费观看| 国产成人高清精品免费软件 | 豆国产96在线|亚洲| g0g0人体全免费高清大胆视频| 国产又大又粗又硬又长免费| 国产AV无码专区亚洲AV蜜芽 | 亚洲中文字幕无码专区| 国产成人亚洲精品蜜芽影院| 日本v片免费一区二区三区| 亚洲日本VA午夜在线电影| 日韩成人免费在线| 菠萝菠萝蜜在线免费视频| 日韩精品内射视频免费观看| 亚洲激情中文字幕| 亚洲精品视频在线观看免费| 亚洲精品高清一二区久久| 国产区在线免费观看| 亚洲AV日韩AV鸥美在线观看| AV激情亚洲男人的天堂国语| 亚洲精品一级无码中文字幕| 国产福利免费视频| 久久精品国产亚洲AV电影| 日韩免费一区二区三区在线| 亚洲国产成人久久精品软件| 亚洲成a人片在线观看国产| 精品多毛少妇人妻AV免费久久| 亚洲成av人在线视| 免费精品国偷自产在线在线| 老子影院午夜伦不卡亚洲| 中文字幕不卡亚洲 | 国产成人免费网站在线观看| 思思久久99热免费精品6| 午夜视频免费成人| 和老外3p爽粗大免费视频| 在线观看亚洲人成网站| 色视频色露露永久免费观看| 成人av片无码免费天天看| 亚洲mv国产精品mv日本mv| 亚洲精品第一国产综合精品99| 97在线视频免费公开观看|