婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > 為什么Java中繼承多數(shù)是有害的

為什么Java中繼承多數(shù)是有害的

熱門標(biāo)簽:淮南騰訊地圖標(biāo)注 開封便宜外呼系統(tǒng)報價 商丘百應(yīng)電話機器人有沒有效果 黃石智能營銷電銷機器人效果 騰訊地圖標(biāo)注商戶改名注冊入駐 漯河辦理400電話 怎樣把地圖標(biāo)注出來 電話機器人的特色和創(chuàng)新 地圖標(biāo)注人員兼職

  大多數(shù)好的設(shè)計者象躲避瘟疫一樣來避免使用實現(xiàn)繼承(extends 關(guān)系)。實際上80%的代碼應(yīng)該完全用interfaces寫,而不是通過extends。“Java設(shè)計模式”一書詳細(xì)闡述了怎樣用接口繼承代替實現(xiàn)繼承。這篇文章描述設(shè)計者為什么會這么作。

  Extends是有害的;也許對于Charles Manson這個級別的不是,但是足夠糟糕的它應(yīng)該在任何可能的時候被避開。“JAVA設(shè)計模式”一書花了很大的部分討論用interface繼承代替實現(xiàn)繼承。

  好的設(shè)計者在他的代碼中,大部分用interface,而不是具體的基類。本文討論為什么設(shè)計者會這樣選擇,并且也介紹一些基于interface的編程基礎(chǔ)。

  接口(Interface)和類(Class)?

  一次,我參加一個Java用戶組的會議。在會議中,Jams Gosling(Java之父)做發(fā)起人講話。在那令人難忘的QA部分中,有人問他:“如果你重新構(gòu)造Java,你想改變什么?”。“我想拋棄classes”他回答。在笑聲平息后,它解釋說,真正的問題不是由于class本身,而是實現(xiàn)繼承(extends) 關(guān)系。接口繼承(implements關(guān)系)是更好的。你應(yīng)該盡可能的避免實現(xiàn)繼承。

  失去了靈活性

  為什么你應(yīng)該避免實現(xiàn)繼承呢?第一個問題是明確的使用具體類名將你固定到特定的實現(xiàn),給底層的改變增加了不必要的困難。

  在當(dāng)前的敏捷編程方法中,核心是并行的設(shè)計和開發(fā)的概念。在你詳細(xì)設(shè)計程序前,你開始編程。這個技術(shù)不同于傳統(tǒng)方法的形式----傳統(tǒng)的方式是設(shè)計應(yīng)該在編碼開始前完成----但是許多成功的項目已經(jīng)證明你能夠更快速的開發(fā)高質(zhì)量代碼,相對于傳統(tǒng)的按部就班的方法。但是在并行開發(fā)的核心是主張靈活性。你不得不以某一種方式寫你的代碼以至于最新發(fā)現(xiàn)的需求能夠盡可能沒有痛苦的合并到已有的代碼中。

  勝于實現(xiàn)你也許需要的特征,你只需實現(xiàn)你明確需要的特征,而且適度的對變化的包容。如果你沒有這種靈活,并行的開發(fā),那簡直不可能。

  對于Inteface的編程是靈活結(jié)構(gòu)的核心。為了說明為什么,讓我們看一下當(dāng)使用它們的時候,會發(fā)生什么。考慮下面的代碼:

  f()
  {
  LinkedList list = new LinkedList();
  //...
  g( list );
  }

  g( LinkedList list )
  {
  list.add( ... );
  g2( list )
  }

  假設(shè)一個對于快速查詢的需求被提出,以至于這個LinkedList不能夠解決。你需要用HashSet來代替它。在已有代碼中,變化不能夠局部化,因為你不僅僅需要修改f()也需要修改g()(它帶有LinkedList參數(shù)),并且還有g(shù)()把列表傳遞給的任何代碼。象下面這樣重寫代碼:

  f()
  {
  Collection list = new LinkedList();
  //...
  g( list );
  }

  g( Collection list )
  {
  list.add( ... );
  g2( list )
  }

  這樣修改Linked list成hash,可能只是簡單的用new HashSet()代替new LinkedList()。就這樣。沒有其他的需要修改的地方。

  作為另一個例子,比較下面兩段代碼:

  f()
  {
  Collection c = new HashSet();
  //...
  g( c );
  }

  g( Collection c )
  {
  for( Iterator i = c.iterator(); i.hasNext() )
  do_something_with( i.next() );
  }

  和

  f2()
  {
  Collection c = new HashSet();
  //...
  g2( c.iterator() );
  }

  g2( Iterator i )
  {
  while( i.hasNext() )
  do_something_with( i.next() );
  }


  g2()方法現(xiàn)在能夠遍歷Collection的派生,就像你能夠從Map中得到的鍵值對。事實上,你能夠?qū)慽terator,它產(chǎn)生數(shù)據(jù),代替遍歷一個Collection。你能夠?qū)慽terator,它從測試的框架或者文件中得到信息。這會有巨大的靈活性。

  耦合

  對于實現(xiàn)繼承,一個更加關(guān)鍵的問題是耦合---令人煩躁的依賴,就是那種程序的一部分對于另一部分的依賴。全局變量提供經(jīng)典的例子,證明為什么強耦合會引起麻煩。例如,如果你改變?nèi)肿兞康念愋停敲此杏玫竭@個變量的函數(shù)也許都被影響,所以所有這些代碼都要被檢查,變更和重新測試。而且,所有用到這個變量的函數(shù)通過這個變量相互耦合。也就是,如果一個變量值在難以使用的時候被改變,一個函數(shù)也許就不正確的影響了另一個函數(shù)的行為。這個問題顯著的隱藏于多線程的程序。

  作為一個設(shè)計者,你應(yīng)該努力最小化耦合關(guān)系。你不能一并消除耦合,因為從一個類的對象到另一個類的對象的方法調(diào)用是一個松耦合的形式。你不可能有一個程序,它沒有任何的耦合。然而,你能夠通過遵守OO規(guī)則,最小化一定的耦合(最重要的是,一個對象的實現(xiàn)應(yīng)該完全隱藏于使用他的對象)。例如,一個對象的實例變量(不是常量的成員域),應(yīng)該總是private。我意思是某段時期的,無例外的,不斷的。(你能夠偶爾有效地使用protected方法,但是protected實例變量是可憎的事)同樣的原因你應(yīng)該不用get/set函數(shù)---他們對于是一個域公用只是使人感到過于復(fù)雜的方式(盡管返回修飾的對象而不是基本類型值的訪問函數(shù)是在某些情況下是由原因的,那種情況下,返回的對象類是一個在設(shè)計時的關(guān)鍵抽象)。

  這里,我不是書生氣。在我自己的工作中,我發(fā)現(xiàn)一個直接的相互關(guān)系在我OO方法的嚴(yán)格之間,快速代碼開發(fā)和容易的代碼實現(xiàn)。無論什么時候我違反中心的OO原則,如實現(xiàn)隱藏,我結(jié)果重寫那個代碼(一般因為代碼是不可調(diào)試的)。我沒有時間重寫代碼,所以我遵循那些規(guī)則。我關(guān)心的完全實用?我對干凈的原因沒有興趣。

   脆弱的基類問題

  現(xiàn)在,讓我們應(yīng)用耦合的概念到繼承。在一個用extends的繼承實現(xiàn)系統(tǒng)中,派生類是非常緊密的和基類耦合,當(dāng)且這種緊密的連接是不期望的。設(shè)計者已經(jīng)應(yīng)用了綽號“脆弱的基類問題”去描述這個行為。基礎(chǔ)類被認(rèn)為是脆弱的是,因為你在看起來安全的情況下修改基類,但是當(dāng)從派生類繼承時,新的行為也許引起派生類出現(xiàn)功能紊亂。你不能通過簡單的在隔離下檢查基類的方法來分辨基類的變化是安全的;而是你也必須看(和測試)所有派生類。而且,你必須檢查所有的代碼,它們也用在基類和派生類對象中,因為這個代碼也許被新的行為所打破。一個對于基礎(chǔ)類的簡單變化可能導(dǎo)致整個程序不可操作。

  讓我們一起檢查脆弱的基類和基類耦合的問題。下面的類extends了Java的ArrayList類去使它像一個stack來運轉(zhuǎn):

  class Stack extends ArrayList
  {
  private int stack_pointer = 0;

  public void push( Object article )
  {
  add( stack_pointer++, article );
  }

  public Object pop()
  {
  return remove( --stack_pointer );
  }

  public void push_many( Object[] articles )
  {
  for( int i = 0; i articles.length; ++i )
   push( articles[i] );
  }
  }

  甚至一個象這樣簡單的類也有問題。思考當(dāng)一個用戶平衡繼承和用ArrayList的clear()方法去彈出堆棧時:

  Stack a_stack = new Stack();
  a_stack.push("1");
  a_stack.push("2");
  a_stack.clear();

  這個代碼成功編譯,但是因為基類不知道關(guān)于stack指針堆棧的情況,這個stack對象當(dāng)前在一個未定義的狀態(tài)。下一個對于push()調(diào)用把新的項放入索引2的位置。(stack_pointer的當(dāng)前值),所以stack有效地有三個元素-下邊兩個是垃圾。(Java的stack類正是有這個問題,不要用它).

  對這個令人討厭的繼承的方法問題的解決辦法是為Stack覆蓋所有的ArrayList方法,那能夠修改數(shù)組的狀態(tài),所以覆蓋正確的操作Stack指針或者拋出一個例外。(removeRange()方法對于拋出一個例外一個好的候選方法)。

  這個方法有兩個缺點。第一,如果你覆蓋了所有的東西,這個基類應(yīng)該真正的是一個interface,而不是一個class。如果你不用任何繼承方法,在實現(xiàn)繼承中就沒有這一點。第二,更重要的是,你不能夠讓一個stack支持所有的ArrayList方法。例如,令人煩惱的removeRange()沒有什么作用。唯一實現(xiàn)無用方法的合理的途徑是使它拋出一個例外,因為它應(yīng)該永遠(yuǎn)不被調(diào)用。這個方法有效的把編譯錯誤成為運行錯誤。不好的方法是,如果方法只是不被定義,編譯器會輸出一個方法未找到的錯誤。如果方法存在,但是拋出一個例外,你只有在程序真正的運行時,你才能夠發(fā)現(xiàn)調(diào)用錯誤。

  對于這個基類問題的一個更好的解決辦法是封裝數(shù)據(jù)結(jié)構(gòu)代替用繼承。這是新的和改進(jìn)的Stack的版本:

  class Stack
  {
  private int stack_pointer = 0;
  private ArrayList the_data = new ArrayList();

  public void push( Object article )
  {
  the_data.add( stack_poniter++, article );
  }

  public Object pop()
  {
  return the_data.remove( --stack_pointer );
  }

  public void push_many( Object[] articles )
  {
  for( int i = 0; i o.length; ++i )
   push( articles[i] );
  }
  }

  到現(xiàn)在為止,一直都不錯,但是考慮脆弱的基類問題,我們說你想要在stack創(chuàng)建一個變量, 用它在一段周期內(nèi)跟蹤最大的堆棧尺寸。一個可能的實現(xiàn)也許象下面這樣:

  class Monitorable_stack extends Stack
  {
  private int high_water_mark = 0;
  private int current_size;

  public void push( Object article )
  {
  if( ++current_size > high_water_mark )
   high_water_mark = current_size;
   super.push( article );
  }

  publish Object pop()
  {
  --current_size;
  return super.pop();
  }

  public int maximum_size_so_far()
  {
  return high_water_mark;
  }
  }

  這個新類運行的很好,至少是一段時間。不幸的是,這個代碼發(fā)掘了一個事實,push_many()通過調(diào)用push()來運行。首先,這個細(xì)節(jié)看起來不是一個壞的選擇。它簡化了代碼,并且你能夠得到push()的派生類版本,甚至當(dāng)Monitorable_stack通過Stack的參考來訪問的時候,以至于high_water_mark能夠正確的更新。

標(biāo)簽:武威 大興安嶺 鄭州 亳州 紅河 馬鞍山 拉薩 岳陽

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《為什么Java中繼承多數(shù)是有害的》,本文關(guān)鍵詞  為什么,Java,中,繼承,多數(shù),;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《為什么Java中繼承多數(shù)是有害的》相關(guān)的同類信息!
  • 本頁收集關(guān)于為什么Java中繼承多數(shù)是有害的的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    国产成人精品影院| 欧美精三区欧美精三区| 亚洲h精品动漫在线观看| ww亚洲ww在线观看国产| 欧美亚洲图片小说| 韩国av一区二区三区四区 | 欧美精品一二三四| heyzo一本久久综合| 精品一区二区三区视频在线观看| 五月天婷婷综合| 亚洲综合色自拍一区| 亚洲少妇屁股交4| xnxx国产精品| 精品成人一区二区三区四区| www国产精品av| 成人h版在线观看| 日韩av中文字幕一区二区| 亚洲人成精品久久久久| 国产精品卡一卡二卡三| 久久老女人爱爱| 久久视频一区二区| 欧美精品一区男女天堂| 26uuu成人网一区二区三区| 一区二区三区中文在线观看| 国产精品传媒在线| 国产精品美女久久久久aⅴ国产馆 国产精品美女久久久久av爽李琼 国产精品美女久久久久高潮 | 久久99久久久久| 亚洲精品在线观看网站| 91同城在线观看| 99国产精品视频免费观看| 视频一区国产视频| 午夜免费久久看| 舔着乳尖日韩一区| 午夜视频一区二区| 性欧美大战久久久久久久久| 亚洲成a人v欧美综合天堂| 无吗不卡中文字幕| 午夜精品久久一牛影视| 免费看黄色91| 国产成人亚洲综合a∨婷婷图片| 国产大陆亚洲精品国产| 99精品视频一区| 欧美亚洲综合久久| 日韩欧美在线观看一区二区三区| 欧美午夜精品一区二区三区| 国产精品888| 久久国产精品99精品国产| 精品一区二区三区日韩| 成人午夜在线免费| 欧美日韩精品系列| 亚洲精品在线网站| 国产精品久久一级| 日韩二区三区四区| 风间由美一区二区三区在线观看 | 精品国产乱码久久久久久久| 国产午夜亚洲精品午夜鲁丝片| 国产精品青草久久| 亚洲国产日韩综合久久精品| 久久99在线观看| 蜜臂av日日欢夜夜爽一区| 丝袜美腿亚洲综合| 韩国av一区二区三区| 成人性生交大片免费看在线播放| 欧洲日韩一区二区三区| 国产亚洲欧美色| 亚洲一区二区三区中文字幕在线| 狠狠狠色丁香婷婷综合激情| 欧美怡红院视频| 日本一区二区视频在线| 三级一区在线视频先锋 | 国产成人免费9x9x人网站视频| 一本大道久久a久久综合| 久久久五月婷婷| 亚洲国产美国国产综合一区二区| 成人在线视频一区| 精品福利av导航| 日韩avvvv在线播放| 91看片淫黄大片一级在线观看| 2021中文字幕一区亚洲| 免费观看91视频大全| 欧美专区亚洲专区| 亚洲视频一区在线| 成人av免费在线| 国产色产综合产在线视频| 美国欧美日韩国产在线播放| 欧美日韩一区二区三区四区五区| 中文字幕一区二区三区在线观看 | 国产三区在线成人av| 中文字幕国产一区| 久久99精品久久久久| 欧美区在线观看| 性久久久久久久| 欧美女孩性生活视频| 亚洲一区自拍偷拍| 在线观看欧美黄色| 亚洲乱码国产乱码精品精可以看| 免费看欧美女人艹b| 国产99久久久国产精品潘金网站| 中文字幕av一区二区三区| 色8久久人人97超碰香蕉987| 日韩av不卡一区二区| 久久一区二区视频| 色妹子一区二区| 蜜臀91精品一区二区三区| 国产精品少妇自拍| 欧美日本乱大交xxxxx| 国产日韩欧美不卡在线| 99在线精品观看| 亚洲国产综合人成综合网站| 日韩欧美一级二级三级久久久| 国产精品一区二区免费不卡| 亚洲欧美综合色| 日韩一级二级三级精品视频| www.欧美.com| 九色porny丨国产精品| 一区二区三区不卡视频| 日韩免费一区二区| 91国偷自产一区二区三区观看 | 17c精品麻豆一区二区免费| 91精品中文字幕一区二区三区| 丁香六月综合激情| 五月天激情综合| 国产精品久久午夜夜伦鲁鲁| 自拍偷拍国产亚洲| 91美女在线观看| 精品一区二区三区久久| 亚洲午夜激情网页| 中文字幕一区二区三区乱码在线| 国产风韵犹存在线视精品| 免费久久99精品国产| 亚洲国产一区二区三区青草影视| 久久青草国产手机看片福利盒子| 国产精品夜夜嗨| 日韩精品五月天| 亚洲午夜av在线| 亚洲免费大片在线观看| 国产精品视频第一区| 国产亚洲一区二区三区四区| 欧美不卡在线视频| 日韩一区二区精品在线观看| 欧美猛男gaygay网站| 色94色欧美sute亚洲线路二| 91丨九色丨蝌蚪丨老版| 99久久综合国产精品| 成人短视频下载| av在线不卡电影| 色婷婷国产精品久久包臀| 色婷婷久久久综合中文字幕| 欧美亚洲综合在线| 久久精品国产亚洲高清剧情介绍| 亚洲高清免费一级二级三级| 一区二区三区中文字幕在线观看| 国产精品久久久久一区二区三区 | 色综合久久久久网| 99国产精品99久久久久久| 99精品欧美一区| 91久久国产最好的精华液| 欧美写真视频网站| 欧美日韩视频不卡| 欧美一区二区三区精品| 日韩欧美一区二区视频| 精品成人在线观看| 国产精品乱码妇女bbbb| 亚洲激情六月丁香| 午夜成人免费视频| 黄色日韩三级电影| 欧美不卡一区二区三区| 欧美一区二区在线免费播放| 欧美电视剧免费观看| 精品毛片乱码1区2区3区| 久久色视频免费观看| 中文字幕日韩一区| 日韩avvvv在线播放| 国产精品888| 欧美日韩精品福利| 精品国产第一区二区三区观看体验| 国产欧美日韩精品a在线观看| 亚洲欧洲综合另类在线| 日韩中文字幕91| 国产精品66部| 欧美性淫爽ww久久久久无| 精品久久久三级丝袜| 中文字幕日韩欧美一区二区三区| 亚洲成人精品一区二区| 国产美女娇喘av呻吟久久| 在线观看视频一区| 久久久精品蜜桃| 亚洲欧美激情视频在线观看一区二区三区| 亚洲成人777| 成人av网站在线观看| 欧美区视频在线观看| 99免费精品在线观看| 国产亚洲欧美日韩日本| 欧美国产一区视频在线观看| 一二三四区精品视频| 激情小说欧美图片| 3d成人h动漫网站入口| 亚洲日本在线天堂| 国产在线一区二区| 69精品人人人人|