0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

你是使用函數(shù)式編程還是面向?qū)ο缶幊谭绞剑?/h1>

作為程序員,你是使用函數(shù)式編程還是面向?qū)ο缶幊谭绞剑?br />
在本文中,擁有 10 多年軟件開發(fā)經(jīng)驗的作者從面向?qū)ο缶幊痰娜筇匦浴^承、封裝、多態(tài)三大角度提出了自己的疑問,并深刻表示是時候和面向?qū)ο缶幊陶f再見了。

幾十年來我都在用面向?qū)ο蟮恼Z言編程。我用過的第一個面向?qū)ο蟮恼Z言是 C++,后來是 Smalltalk,最后是 .NET 和 Java。 我曾經(jīng)對使用繼承、封裝和多態(tài)充滿熱情。它們是范式的三大支柱。 我渴望實現(xiàn)重用之美,并在這個令人興奮的新天地中享受前輩們積累的智慧。 想到將現(xiàn)實世界的一切映射到類中,使得整個世界都可以得到整齊的規(guī)劃,我無法抑制自己的興奮。 然而我大錯特錯了。
01繼承,倒塌的第一根支柱 乍一看,繼承似乎是面向?qū)ο蠓妒降淖畲髢?yōu)勢。所有新手教程講解繼承時都會拿出最簡單的繼承的例子,而這個例子似乎很符合邏輯。

然后就是滿篇的重用了。甚至以后的一切都是重用了。 我囫圇吞下這一切,然后帶著新發(fā)現(xiàn)興沖沖地奔向世界了。香蕉猴子叢林問題帶著滿腔的信仰和解決問題的熱情,我開始構(gòu)建類的層次結(jié)構(gòu)然后寫代碼。似乎一切皆在掌控中。 我永遠(yuǎn)不會忘記我準(zhǔn)備從已有的類繼承并實現(xiàn)重用的那一天。那是我期待已久的時刻。 后來有了新的項目,我想起了另一個項目里我很喜歡的那個類。 沒問題,重用拯救一切。我只需要把那個類拿過來用就好了。 嗯……其實……不僅是那一個類。還得把父類也拿過來。但……應(yīng)該就可以了吧。 額……不對,似乎還需要父類的父類……還有……嗯,我們需要所有的祖先類。好吧好吧……搞定了。沒問題。 不錯。但編譯不過,怎么回事?哦我知道了……這個對象還需要另一個對象。所以那個也得拿過來。沒問題…… 等等……我不僅需要那個對象,還需要那個對象的父類,和父類的父類,和……包含的所有對象的所有祖先…… 唉…… Erlang 的創(chuàng)建者 JoeArmstrong 有句名言:

面向?qū)ο笳Z言的問題在于,它們依賴于特定的環(huán)境。你想要個香蕉,但拿到的卻是拿著香蕉的猩猩,乃至最后你擁有了整片叢林。

香蕉猴子叢林的解決方法這個問題的解決方法是,不要把類層次建得那么深。但如果繼承是重用的關(guān)鍵,那么給繼承機(jī)制添加的任何限制都會限制重用。對吧? 沒錯。 那我們可憐的面向?qū)ο蟪绦騿T該怎么辦?指望一杯三聚氰胺奶維系我們的健康嗎? 答案就是:包含和委托(Contain and Delegate)。一會兒會詳細(xì)解釋。菱形繼承問題早晚你會遇到下面這種惡心的問題,有些語言甚至根本解決不了。

大多數(shù)面向?qū)ο笳Z言都不支持這種情況,盡管看上去似乎很符合邏輯。為什么面向?qū)ο笳Z言支持這種情況如此困難? 來看看下面的偽代碼:

ClassPoweredDevice{ } ClassScannerinheritsfromPoweredDevice{ functionstart(){ } } ClassPrinterinheritsfromPoweredDevice{ functionstart(){ } } ClassCopierinheritsfromScanner,Printer{ } 注意 Scanner 和 Printer 類都實現(xiàn)了名為 start 方法。 那么問題來了,Copier繼承哪個start?是Scanner的還是Printer的?肯定不可能同時繼承啊。菱形繼承的解決解決方案很簡單:不要這樣做。 沒錯。大多數(shù)面向?qū)ο蠖疾蛔屇氵@么干。 但是,但是……要是必須這樣建模該怎么辦?我需要重用! 那就必須使用包含和委托。ClassPoweredDevice{ } ClassScannerinheritsfromPoweredDevice{ functionstart(){ } } ClassPrinterinheritsfromPoweredDevice{ functionstart(){ } } ClassCopier{ Scannerscanner Printerprinter functionstart(){ printer.start() } } 注意現(xiàn)在 Copier 類包含一個 Printer 實例和一個 Scanner 實例。然后將 start 函數(shù)委托給 Printer 類的實現(xiàn)。要委托給 Scanner 也很簡單。 這個問題是繼承這根支柱上的另一條裂縫。脆弱的基類問題好吧,那我盡量使用較淺的類層次結(jié)構(gòu),并保證里面沒有環(huán),這樣就不會出現(xiàn)菱形繼承了。 似乎一切都解決了。直到我們發(fā)現(xiàn)…… 我前一天工作得好好的代碼今天出錯了!關(guān)鍵是,我沒有改任何代碼! 嗯也許是個 bug……但等等……的確有些改動…… 但改動的不是我的代碼。似乎改動來自我繼承的那個類。 為什么基類的改動會破壞我的代碼? 原來是這樣…… 看看下面這個基類(用Java寫的,但就算你不懂Java,應(yīng)該也很容易看懂):importjava.util.ArrayList; publicclassArray { privateArrayLista=newArrayList(); publicvoidadd(Objectelement) { a.add(element); } publicvoidaddAll(Objectelements[]) { for(inti=0;i
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 封裝
    +關(guān)注

    關(guān)注

    125

    文章

    7593

    瀏覽量

    142147
  • 編程
    +關(guān)注

    關(guān)注

    88

    文章

    3521

    瀏覽量

    93276
  • C++
    C++
    +關(guān)注

    關(guān)注

    21

    文章

    2085

    瀏覽量

    73304

原文標(biāo)題:面向?qū)ο缶幊?,再見?/p>

文章出處:【微信號:mcu168,微信公眾號:硬件攻城獅】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    工業(yè)機(jī)器人常用的編程方式

    工業(yè)機(jī)器人的編程是確保機(jī)器人能夠執(zhí)行特定任務(wù)的關(guān)鍵步驟。編程方式多種多樣,每種方式都有其特定的應(yīng)用場景和優(yōu)勢。 工業(yè)機(jī)器人編程概述 工業(yè)機(jī)器
    的頭像 發(fā)表于 09-04 09:26 ?185次閱讀

    倉頡來啦,辟謠:不是中文編程

    打造友好開發(fā)體驗和卓越程序性能。 而它的高效編程,安全可靠,輕松并發(fā),卓越性能又為開發(fā)者帶來不一樣的體驗。 特點 高效編程:倉頡支持面向對象、函數(shù)
    發(fā)表于 09-03 08:50

    簡述socket編程中的常用函數(shù)

    Socket編程是一種基于TCP/IP協(xié)議的網(wǎng)絡(luò)編程技術(shù),它允許應(yīng)用程序通過網(wǎng)絡(luò)進(jìn)行通信。在Socket編程中,有許多常用的函數(shù),它們用于創(chuàng)建、配置、連接、發(fā)送和接收數(shù)據(jù)等操作。以下是
    的頭像 發(fā)表于 08-16 10:49 ?153次閱讀

    如何理解socket編程接口

    Socket)和數(shù)據(jù)報Socket(Datagram Socket)。 1.2.1 流式Socket 流式Socket是一種面向連接的通信方式,它在通信雙
    的頭像 發(fā)表于 08-16 10:48 ?220次閱讀

    PLC的編程方式編程語言

    在工業(yè)自動化領(lǐng)域,PLC(Programmable Logic Controller,可編程邏輯控制器)因其強(qiáng)大的控制功能和靈活的編程方式而得到了廣泛應(yīng)用。PLC的編程
    的頭像 發(fā)表于 06-27 14:08 ?416次閱讀

    如何提升嵌入編程能力?

    如何提升嵌入編程能力? 要提升嵌入編程的能力,可以從以下幾點學(xué)習(xí): 1. 理解硬件:熟悉正在編程
    發(fā)表于 06-21 10:01

    三菱fx系列plc支持哪種編程方式

    三菱FX系列PLC是一種廣泛應(yīng)用于工業(yè)自動化領(lǐng)域的可編程邏輯控制器。它具有強(qiáng)大的功能和靈活性,支持多種編程方式,以滿足不同用戶的需求。本文將詳細(xì)介紹三菱FX系列PLC支持的編程
    的頭像 發(fā)表于 06-20 10:24 ?485次閱讀

    基于C/C++面向對象方式封裝socket通信類流程簡析

    在掌握了基于 TCP 的套接字通信流程之后,為了方便使用,提高編碼效率,可以對通信操作進(jìn)行封裝,本著有淺入深的原則,先基于 C 語言進(jìn)行面向過程的函數(shù)封裝,然后再基于 C++ 進(jìn)行面向對象
    的頭像 發(fā)表于 12-26 10:00 ?1457次閱讀

    基于C/C++面向對象方式封裝socket通信類

    在掌握了基于 TCP 的套接字通信流程之后,為了方便使用,提高編碼效率,可以對通信操作進(jìn)行封裝,本著有淺入深的原則,先基于 C 語言進(jìn)行面向過程的函數(shù)封裝,然后再基于 C++ 進(jìn)行面向對象
    的頭像 發(fā)表于 12-26 09:57 ?1141次閱讀

    python窗口圖形界面編程

    Python是一種功能強(qiáng)大而又簡單易用的編程語言,支持多種編程范式,包括面向過程、面向對象函數(shù)
    的頭像 發(fā)表于 11-22 14:23 ?738次閱讀

    淺談C語言面向對象編程思想

    C語言是一種面向過程的語言,但是也可以用結(jié)構(gòu)體和函數(shù)指針來模擬面向對象的特性,比如封裝、繼承和多態(tài)。
    發(fā)表于 11-02 12:27 ?796次閱讀

    MCU上面可以實現(xiàn)面向對象編程嗎?

    為什么都是用的面向過程編程
    發(fā)表于 10-24 07:39

    Java中的Reactive編程示例

    相信響應(yīng)編程經(jīng)常會在各種地方被提到。本篇就為大家從函數(shù)編程一直到Spring WeFlux做一次簡單的講解,并給出一些示例,希望大家可以
    的頭像 發(fā)表于 10-08 16:06 ?842次閱讀
    Java中的Reactive<b class='flag-5'>編程</b>示例

    Python的函數(shù)編程介紹

    代碼的時候,拆成不同的函數(shù),可以大大的提高代碼易讀性。 函數(shù)編程 函數(shù)
    的頭像 發(fā)表于 09-27 16:54 ?1020次閱讀

    Python編程實用指南

    Python 是一種解釋型、面向對象、動態(tài)數(shù)據(jù)類型的高級程序設(shè)計語言。通過 Python 編程,我們能夠解決現(xiàn)實生活中的很多任務(wù)。本書是一本面向實踐的 Python
    發(fā)表于 09-27 06:21