面向對象編程會被拋棄嗎?這五大問題不容忽視

2021-02-13 機器之心

20 世紀 60 年代,編程遇到了一個大問題:計算機還沒有那麼強大,需要以某種方式平衡數據結構和程序之間的能力。這意味著,如果你有大量數據,那麼不將計算機推向極限就無法充分利用這些數據。另外,如果你需要做很多事情,那麼你就不能使用過多的數據,否則計算機將會一直運行下去。接下來到了 1966、1967 年,Alan Kay 從理論上證明可以使用封裝的微型計算機。這些微型計算機不共享數據,而是通過消息傳遞進行通信。這樣就可以更加經濟地使用計算資源。儘管這個想法很巧妙,但直到 1981 年,面向對象編程才成為主流。在那之後,它就沒有停止過吸引新的和經驗豐富的軟體開發者。面向對象的程式設計師市場一如既往地忙碌。但是在最近幾年中,這種已有幾十年歷史的編程範式受到越來越多的批評。難道是在面向對象編程大行其道 40 年之後,技術已經超越了這種範式?

https://towardsdatascience.com/why-developers-are-falling-in-love-with-functional-programming-13514df4048e面向對象編程的主要思想非常簡單:嘗試將一個功能強大的程序整體分解為功能同樣強大的多個部分。這樣就可以將一些數據和那些只在相關數據上使用的函數耦合起來。注意,這僅涵蓋封裝的概念。也就是說,位於對象內部的數據和函數對於外部是不可見的。我們只能通過消息(通常通過 getter 和 setter 函數)與對象的內容進行交互。繼承性和多態性並沒有包含在最初的設計想法中,但是對於現在的面向對象編程而言是必需的。繼承基本上意味著開發者可以定義具有其父類所有屬性的子類。直到 1976 年,即面向對象的程序設計的概念問世十年之後,繼承性才被引入。又過了十年,多態性才進入面向對象的編程。簡單來講,這意味著某種方法或對象可以用做其他方法或對象的模板。從某種意義上說,多態性是繼承性的泛化,因為並不是原始方法或對象的所有屬性都需要傳輸到新實體。相反,你還可以選擇重寫一些屬性。多態性的特殊之處在於,即使兩個實體在原始碼中互相依賴,被調用實體的工作方式也更像插件。這使得開發人員的工作變得輕鬆,因為他們不必擔心運行時的依賴關係。值得一提的是,繼承性和多態性並不是面向對象編程所特有的。真正的區別在於封裝數據及其包含的方法。在計算資源比今天稀缺得多的時代,這是一個天才的想法。

面向對象的編程一經問世,便改變了開發人員看待代碼的方式。20 世紀 80 年代以前,過程式編程非常面向機器。開發人員需要非常了解計算機的工作原理才能編寫好的代碼。通過封裝數據和其他方法,面向對象的編程使軟體開發更加以人為中心,符合人類的直覺。比如,方法 drive() 屬於 car 數據組,而不是 teddybear 組。之後出現的繼承性也很直觀。比如,現代汽車(Hyundai)是汽車的一個子類,並且具有相同的屬性,但 PooTheBear 不是,這樣很好理解。想像一下,你正在設置一個新程序,並且正在考慮設計一個新類。然後,你回想起為另一個項目創建的簡潔的小類,發現其對正在進行的工作很合適。這裡有一個問題:這個類可能是另一個類的子類,因此你需要將它的父類也包含在內。然後你會發現,這個父類可能也是另一個類的子類,以此類推,最後要面對一堆代碼。Erlang 的創建者 Joe Armstrong 曾有一句名言:「面向對象語言的問題在於,它們自帶其自身周圍的所有隱式環境。你想要香蕉,但是得到的卻是拿著香蕉的大猩猩和整個叢林。」這幾乎可以說明一切。復用類是可以的,實際上這可能是面向對象編程的主要優點,但不要將其發揮到極致。有時你應該建立一個新的類,而不是添加大量依賴項。

想像一下,如果你已經成功地將另一個項目中的類復用於新的代碼,那麼如果基類發生變化會怎樣?這可能會破壞你整個新項目的代碼,即使你可能什麼也沒做。一旦有人更改了基類中的一個細節,而這一點又對你的項目至關重要,那麼這種影響將是非常大並且突然的。使用繼承的次數越多,潛在的維護工作就越多。因此,即使在短期內復用代碼非常有效,但從長遠來看,它可能讓你付出一定的代價。利用繼承可以將一類中的屬性傳遞給其他類。但是,如果你想混合兩個不同類的屬性怎麼辦?沒錯,這無法完成,至少常規的方法都不行。以 Copier 類為例(在此引用以下連結文章中的例子:https://medium.com/@cscalfani/goodbye-object-oriented-programming-a59cda4c0e53),Copier 將掃描文件的內容並將其列印在白紙上。那麼它應該是 Scanner 還是 Printer 的子類?這個問題根本沒有完美的答案。即使這個問題不會破壞你的代碼,但它經常出現,會讓人很沮喪。在菱形繼承問題中,Copier 是哪個類的子類是問題的關鍵所在。但或許有個投機取巧的方案:假設 Copier 是父類,Scanner 和 Printer 是僅繼承屬性子集的子類,那麼問題就解決了。但如果你的 Copier 是黑白的,而 Printer 也能夠處理彩色,那怎麼辦?從這個意義上說,Printer 不是 Copier 的一種泛化嗎?如果 Printer 連接了 WiFi,而 Copier 沒有呢?類上堆積的屬性越多,建立適當的層次結構就越困難。在你所處理的屬性集群中,Copier 共享了 Printer 的一些屬性,但不是全部屬性,反之亦然。在大型複雜項目中,層次結構的問題會導致很大的混亂。

你可能會想到進行沒有層次結構的面向對象編程。我們可以使用屬性集群,並根據需要繼承、擴展或重寫屬性。也許這有點混亂,但這將是對當前問題的準確表示。這裡只存在一個問題:封裝的全部目的是使數據片段彼此之間保持安全,從而使計算效率更高,但沒有嚴格的層次結構,這是行不通的。假設一個對象 A 通過與另一個對象 B 交互來覆蓋層次結構,會發生什麼情況?其他關係的情況並不重要,但當 B 不是 A 的直接父類時,A 必須包含 B 的全部私有引用,否則,它們將無法交互。但是,如果 A 包含 B 的子類也具有的信息,那麼就可以在多個位置修改該信息。因此,有關 B 的信息已經不再安全,並且封裝已經被破壞。儘管許多面向對象的程式設計師都使用這種架構來構建程序,但這並不是面向對象編程,只是一團糟。以上 5 個問題的共同點是它們都存在不合適的繼承。由於繼承沒有包含在面向對象編程的原始形式中,所以這些問題可能不能稱為面向對象本身的問題。但是也並不是只有面向對象編程會被誇大。在純粹的函數式編程中,處理用戶的輸入或在屏幕上輸出消息極其困難。對此,面向對象或面向過程編程會好很多。但仍然有一些開發人員試圖將這些東西用純函數的方式實現,並且編寫幾十行沒人能看懂的代碼。而使用另一種範式就能夠輕鬆地將代碼簡化為幾行可讀的代碼。毫無疑問,函數式編程正在得到更多關注,而面向對象編程近幾年遭到一些詬病。了解新的編程範式並在適當的時候使用它們是很有意義的。無論哪種編程範式,都不需要只遵循一種,在適當的時候使用不同的編程範式才能更好地解決問題。

https://medium.com/madhash/what-is-better-functional-programming-or-object-oriented-9a116c704420面對越來越多的問題,函數式編程可能是更有效的一種選擇。數據分析、機器學習、並行編程,這些領域你投入的越多,你就會越喜歡函數式編程。但是目前面向對象開發的程式設計師的崗位需求量依然比函數式編程開發程式設計師多得多。但是這也並不意味著你不能成為後者,函數式編程開發的程式設計師目前仍然比較稀缺。最有可能的情況是,面向對象的編程將會繼續存在十年左右。當然,選擇相對前衛的方式是好的,但這並不意味著你應該放棄面向對象編程。所以在接下來的幾年中,不要完全放棄它,但至少確保它不是你唯一掌握的程序設計方式。原文連結:https://towardsdatascience.com/object-oriented-programming-is-dead-wait-really-db1f1f05cc44

10月14日,AWS解決方案架構師賀瀏璐將帶來一場live coding,演示如何利用AWS雲服務構建一個簡單的數據管道從爬取、處理到分析視頻內容。

識別二維碼或點擊閱讀原文,立即預約直播。


© THE END 

轉載請聯繫本公眾號獲得授權

投稿或尋求報導:content@jiqizhixin.com

相關焦點

  • 再見了,面向對象的編程
    我可以通過不建立如此複雜的分層來解決這個問題。但是如果繼承是代碼再利用的關鍵,那麼我在該機制上設定的任何限制都將限制代碼再利用的功能。對嗎?對。那麼,一個虔誠的、可憐的OO程式設計師,應該做什麼呢?包含和代理。以後再說。不久的將來,以下問題會抬起它醜陋的、(依語言而定)無解的頭。
  • TIA Portal面向對象編程入門
    儘管時至今日依然有少數人質疑面向對象的編程思想,但我們看到的是面向對象技術發展的越來越好,無論是後端語言(JAVA、C#)或者前端語言(JavaScript、TypeScript),無一不是完全的支持面向對象技術。現在高校的PLC教材基本上採用的還是五六十年前的編程理念,將PLC定位為傳統繼電器控制的替代,以軟元件、寄存器這種古老落後的概念來講授這一門日新月異的現代工業控制編程技術。
  • 面向對象的Qt編程
    從BOP到OOP基於對象的Qt編程(不推薦)
  • JavaScript 的函數式編程與面向對象編程區別在哪?
    本文通過代碼來看一看JavaScript中函數式編程和面向對象編程的差異。
  • 一步步分析:C語言如何面向對象編程
    這篇文章,我們就來聊聊如何在C語言中利用面向對象的思想來編程。也許你在項目中用不到,但是也強烈建議你看一下,因為我之前在跳槽的時候就兩次被問到這個問題。二、什麼是面向對象編程有這麼一個公式:程序=數據結構+算法。C語言中一般使用面向過程編程,就是分析出解決問題所需要的步驟,然後用函數把這些步驟一步一步調用,在函數中對數據結構進行處理(執行算法),也就是說數據結構和算法是分開的。
  • 面向對象編程已死,OOP 永存!
    這種「模式」經常會導致繼承的過度使用,而不會提及過度使用繼承,其實違反了OOP(OOP,Object Oriented Programming,面向對象編程,是一種計算機編程架構)原則。那麼如何避免這種情況呢?本文作者,會給大家介紹下真正的設計指南。
  • 史上最全 Python 面向對象編程
    .html面向對象編程和函數式編程(面向過程編程)都是程序設計的方法,不過稍有區別。面向對象編程:1. 導入各種外部庫2. 設計各種全局變量3. 決定你要的類4. 給每個類提供完整的一組操作5. 明確地使用繼承來表現不同類之間的共同點6.
  • 【譯】使用UIKit進行面向對象的編程
    毫無疑問,協議擴展將是Swift這道菜中的一位重要調料。Apple甚至建議儘可能的使用協議(protocol)來替換類(class)--這是面向協議編程的關鍵。我讀過許多文章,其中對協議擴展的定義講的很清晰。但都沒有說明面向協議編程真正能為UI開發帶來些什麼。當前可用的一些示例代碼並不是基於一些實際場景的,而且沒用應用任何框架。
  • 【編程基礎第五講】java面向對象思想如何理解?
    存在的疑惑:如何理解面向對象的思想?
  • 是時候讓 JavaScript 面向對象了!
    但是從長期來看這很可能造成很多問題。許多 JavaScript 程式設計師都沒有接受過面向對象編程方法的培訓。當然,這也不是他們的錯。有時候我們只會學習工作中用得到的東西,還有的時候連我們自己都不清楚我們缺乏哪方面的知識。面向對象編程的基本思路完全不同。其背後的思想是:你將所需要做的功能抽象成一個「對象」,然後一遍遍地調用這個對象來完成你想要的功能。
  • Java面向對象編程開發認證考試項目介紹
    2、職業定義java是面向對象編程的語言,面向對象編程有四個特點:抽象,封裝,繼承,多態。面向對象區別於bai以前的面向過程,du像Java、vb、vc都是zhi採用面向dao對象編zhuan程,當然更高級的還有面向接口編shu程。
  • C 語言面向對象編程 - 繼承
    點擊上方藍字,關注微聯智控可點擊右上角的 …,分享這篇文章
  • Python面向對象編程指南
    Python面向對象編程指南第1部分 用特殊方法實現Python風格的類第1章 __init__方法1.1 隱式的積累——object
  • 面向對象聖經
    上帝說:「要有Class!把這些重複的方法代碼從對象中剝離出來,放到一個公共的Class中!還有人說:「這不又回到我們原始的狀態去了嗎?  void push(Stack *s,int data)」上帝告誡說:「雖然本質相同,但是表達形式不同,之前你在語法層面必須傳遞一個Stack 對象到函數,現在在語法層面不用加了。我會在編譯後的代碼,或者運行時自動給你們加上。」Guido說:「我還是喜歡把self加到方法上!」
  • DDD 就是把面向對象做好
    這就是由來已久的充血、貧血之爭。充血模型、貧血模型面向對象在處理對象的存儲時,有兩種風格一直爭論不斷。將業務邏輯放到領域對象中,對象不僅需要承載數據也需要承載行為,這種編程邏輯被稱作充血模型。各自的局限性充血模型的好處是可以封裝一些業務邏輯,避免面向模型的開發退化為事務腳本化的代碼。事務腳本化的代碼在前面的文章中已經討論過,這是一種面向功能的開發方法,而非模型的開發方法,會造成代碼中業務邏輯重複,一致性差的問題。因此在 DDD 的相關書籍以及 《企業應用架構》中被反覆提及,編程大師們大多鼓勵使用充血模型。
  • C語言實現面向對象的原理
    雖然我們的教材有這麼一個結論:C語言是面向過程的語言,C++是面向對象的程式語言,但面向對象的概念是在C語言階段就有了,而且應用到了很多地方,比如某些作業系統內核、通信協議等。閱讀文本之前肯定有讀者會問這樣的問題:我們有C++面向對象的語言,為什麼還要用C語言實現面向對象呢?C語言這種非面向對象的語言,同樣也可以使用面向對象的思路來編寫程序的。只是用面向對象的C++語言來實現面向對象編程會更簡單一些,但是C語言的高效性是其他面向對象程式語言無法比擬的。
  • 分享幾點單片機面向對象思想的案例
    為什要記住函數指針,因為在單片機面向對象編程中,結構體的成員不是變量就是函數指針這兩種類型。變量就不用說了,函數指針理解就好。三、為什麼要面向對象?我們知道,現有的編程範式主要是:面向過程編程、面向對象編程、函數式編程。
  • 如何寫一手漂亮的模型:面向對象編程的設計原則綜述
    而採用類的方法來構建模型會令代碼非常具有可讀性和條理性,本文介紹了算法實現中使用類和方法來構建模型所需要注意的設計原則,它們可以讓我們的機器學習代碼更加美麗迷人。大多數現代程式語言都支持並且鼓勵面向對象編程(OOP)。
  • golang 面向對象分析
    說道面向對象(OOP)編程, 就不得不提到下面幾個概念: 抽象, 封裝, 繼承, 多態其實有個問題Is Go An Object Oriented Language?, 隨便谷歌了一下, 你就發現討論這個的文章有很多:那麼問題來了由於之前是寫 C++, 但是說到 Go 面向對象編程, 總是感覺怪怪的, 總感覺缺少點什麼. 我搜集了一些資料和例子, 加上我的一些理解, 整理出這樣一篇文章.一.
  • 面向對象基礎 - Class使用及三大特性
    面向過程與面向對象的區別面向過程 Procedural programming在實際生活中,當我們遇到一個問題需要解決時,我們會進行一系列的工作來解決問題.如:當餓了的時候,需要解決這個問題,我們需要做一系列事情:在計算機編程中,利用各種語言解決各種問題.和在實際生活中一樣,需要一系列的步驟來編寫一個程序.這種編程思想是注重與過程的.所以叫做面向過程編程.