C 語言不敗神話
C 語言不敗神話
資料來源:https://mp.weixin.qq.com/s?__biz=MzUyMzM2ODUwMA==&mid=2247487961&idx=1&sn=9927efcbf47f735a719664282afdfe15&chksm=fa3cf082cd4b7994b12360643d71675f1bbb7cd1e9444ffc28cd78e189290dd026ab47178c0d&scene=126&sessionid=1594283895&key=e6152aaf211ea82498f0df79193997c637d14d8890d93971033cedc1e28cd8642dfabebea26651d402f18221b843bfb729dbf44c58ab2dc56efcd515e85db402bfe78a97be354302c3f358faeb9eb682&ascene=1&uin=MjIwODk2NDgxNw%3D%3D&devicetype=Windows+10+x64&version=62090529&lang=zh_TW&exportkey=AgWPlr4nW3zX6yNIZJvcYYA%3D&pass_ticket=i8S8kN4UMUSvyfU%2FUB66Tf%2FEUPIr%2FrteKKGSiY8bSvQPproOnNfh2D6%2FcI9VbDoK
01、為什麼C語言不會過時
C語言已經把硬件的運行效率壓縮到了極致。
02、為什麼C仍佔據統治地位?
於一種計算機行業的技術來說尤其如此。自1972年誕生以來,C語言一直保持生龍活虎的狀態,時至今日它仍然是我們用來搭建軟件世界的基礎建築材料之一。
但有時一種技術能夠長期存在,只是因為人們還沒有來得及發明新的東西來取代它而已。在過去的幾十年裡,出現了許多其他語言——其中一些明確地被設計用於挑戰C的主導地位,有些語言試圖憑藉自己的人氣慢慢瓦解C語言的統治地位。
為C需要被替換掉的觀點爭辯是簡單的。編程語言研究和軟件開發實踐都暗示瞭如何比C更好地去做事。但歷經數十年的研究和開發,C語言的地位卻依舊穩固。很少有其他語言能夠在性能、裸機兼容性或通用性等方面擊敗它。不過,2018年C是如何與那些明星編程語言競爭的呢,其中細節仍值得一看。
C vs. C ++
當然了,C最常被拿來與C ++進行比較,正如其名稱本身所暗示的那樣,C++作為對C語言的擴展而被創建出來。C ++和C之間的差異可以概括為C++更加廣泛(褒)或更加寬泛(貶),具體取決於這個問題你是問的C還是C++程序員。(笑)
雖然C ++的語法等方面仍然是類C的,但它提供了許多在原生的C中本不可用的非常實用的功能:命名空間(namespace),模板(template),異常(exception),自動內存管理(automatic memory management)等等。需要頂級性能的項目,例如涉及數據庫,機器學習系統的項目通常是用C ++編寫的,以便項目能盡可能地榨取以及利用到每一點性能。
此外,與C相比,C ++在持續地更加積極地擴展。即將推出的C ++ 20會帶來更多功能供開發者享用,包括模塊,協同程序,同步庫,以及概念,這些使模板更易於使用。C standard的最新版本只進行了少量更新,並側重於保持向後兼容性。
事實上,C ++中的所有附加功能同樣也可能成為累贅。而且是很大的累贅。您使用的C ++專屬功能越多,引入的複雜度就越高,對結果的修正就越困難。將自己局限於僅一個C ++子集的開發人員可以避免許多開發中嚴重的坑和額外負擔。但是有些團隊想要從根兒上防範C ++的過度複雜性。堅持使用C能迫使開發人員將自己局限於一個子集。例如,Linux內核開發團隊就直接避開了C ++。
選C而不選C++對您——以及任何將會維護你代碼的開發人員——來說都是可行的,通過採用強制簡約主義來避免與C ++的複雜性糾纏。當然,C ++擁有豐富的高級功能,這是有它自己的道理的。但如果極簡主義更適合當前和未來的項目——以及負責項目的團隊——那麼還是選C更明智一些。
C vs. Java
幾十年了,Java仍然是企業軟件開發的主力軍之一——並且也是寬泛而言的開發的主力軍之一。許多最重要的企業軟件項目都是用Java編寫的——包括絕大多數Apache Software Foundation項目——而Java仍然是開發企業級需求項目的可行語言。
Java的語法從C和C ++中藉鑑了很多東西。但是,與C不同的是,Java默認情況下不會編譯為本機代碼。相反,Java運行時環境,JVM,JIT(實時)編譯Java代碼以在目標環境中運行。在適當的情況下,JIT編譯後的Java代碼可以接近甚至超過C的性能。
Java背後的“一次編寫,隨處運行”的理念也允許Java程序在目標架構上進行相對較少的調整即可運行。相比之下,雖然C已被移植到許多架構中,但任何給定的C程序仍可能需要重新量身定做才能在,打個比方,Windows與Linux,兩種不同的os之間正常運行。
這種可移植性和強大性能的結合,以及龐大的軟件庫和框架組成的生態,使Java成為構建企業應用程序的首選語言。
Java輸給C的地方是一個Java從未打算競爭的領域:靠近底層結構運行,或直接與硬件打交道。C代碼被編譯成機器代碼,由進程直接執行。Java被編譯成字節碼,這是一種隨後會被JVM解釋器轉換為機器代碼的中間代碼。此外,儘管Java的自動內存管理在大多數情況下都是個優點,但C更適合於必須充分利用有限內存資源的情況。
也就是說,在某些方面,Java在速度方面可以接近於C。JVM的JIT引擎在運行時根據程序行為優化例程,允許進行許多類型的優化,而這些優化是在未提前編譯的C中無法實現的。雖然Java運行時自動執行內存管理,但一些較新的應用程序可以解決這個問題。例如,Apache Spark部分地通過使用繞過JVM的自定義內存管理代碼來優化內存中處理。
C vs. C#和.Net
在推出近二十年後,C#和.Net 框架仍然是企業軟件世界的主要組成部分。有人說C#和.Net是微軟對Java的回應——一個託管代碼編譯器系統和通用運行庫——C和Java之間的許多種對比也適用於C和C#或.Net之間。
與Java(以及某種程度上來說Python也是如此)一樣,.Net提供跨各種平台的可移植性和龐大的集成軟件生態系統。考慮到.Net世界中有多少面向企業的開發,這些都是不小的優勢。當您使用C#或任何其他.Net語言開發程序時,您可以使用為.Net運行時編寫的大量工具和庫。
.NET另一個類似Java的優勢是JIT優化。C#和.Net程序可以按照C語言提前編譯,但它們主要由.Net運行時進行即時編譯,並使用運行時信息進行優化。JIT編譯允許對無法在C中執行的運行著的.Net程序進行各種就地優化。
與C一樣,C#和.Net提供各種直接訪問內存的機制。堆,棧和非託管系統內存都可以通過.Net API和對象訪問。開發人員可以使用.Net中的unsafe模式來實現更高的性能。
但這些都不是沒有代價的。託管對象和unsafe對像不能被任意交換,並且它們之間的編組會降低性能。因此,要最大化.Net應用程序的性能需要將託管和非託管對象之間的變動保持在最低限度。
如果您無法承擔託管與非託管內存之間變動造成的性能損失,或者.Net運行時對於目標環境(例如,內核空間)來說是一個糟糕的選擇,或者可能根本不可用,那麼C就是你所需要的。與C#和.Net不同,C被默認可以解鎖對內存的訪問權。
C vs. Go
Go的語法很大程度上借鑒了C——花括號作為定界符,語句以分號結束,等等。精通C的開發人員通常可以毫不費力地直接使用Go,甚至算上Go的獨有功能,如命名空間和包管理,對開發人員來說也並不困難。
代碼可讀性是Go的指導設計目標之一:讓開發人員可以輕鬆掌握任何Go項目,並在短時間內熟練掌握代碼庫。C代碼庫可能很難理解,因為它們很容易聚集大量專屬於某個項目或某個團隊的宏和和#ifdef。Go的語法及其內置的代碼格式以及項目管理工具旨在避免這種結構性問題。
Go還提供了諸如goroutine和channel之類的附加功能,用於處理並發性和組件之間的消息傳遞的語言級別的工具。C需要開發者手動完成或由外部庫提供,但Go提供了開箱即用的這些功能,使得構建需要這些功能的軟件變得更加容易。
Go與C最深層次的不同之處在於內存管理方面。默認情況下,Go的對象會被自動管理並自動進行回收。對於大多數編程工作來說,這非常方便。但這也意味著任何需要確定性處理內存的程序都會更難編寫。
Go確實包含了用於繞過Go的某些類型處理安全性的unsafe包,例如使用Pointer類型讀取和寫入任意內存。但unsafe會附帶一個warning說用它編寫的程序“可能是不可移植的,並且不受Go 1兼容性指南的保護。”
Go非常適合構建命令行實用程序和網絡服務等,因為這些很少用到太過細緻的操作。但是,如果是低級設備驅動程序,內核空間操作系統組件以及其他需要嚴格控制內存佈局和管理的任務,那麼就最好用C來創建。
C vs. Rust
在某些方面,Rust是對C和C ++創建的內存管理難題的回應,也是對這兩種語言的許多其他缺點的回應。Rust編譯為本機機器代碼,因此就性能而言,它被認為與C相當。但默認情況下,內存安全才是Rust的主要賣點。
Rust的語法和編譯規則可幫助開發人員避免常見的內存管理錯誤。如果程序有一個不符合Rust語法的內存管理問題,它就不會被編譯。剛接觸這種語言的新手,特別是以前用C語言的開發者,由於C語言為這類bug提供了充足的容錯空間,所以他們接觸Rust的第一步是學習如何安撫編譯器。但Rust的支持者認為,這種短期的痛苦有一個長期的回報:更安全的,不會減緩速度的代碼。
Rust還通過其工具改進了C語言。默認情況下,項目和組件管理是Rust提供的工具鏈的一部分,與Go相同。有一種默認的,推薦的方法來管理包,組織項目文件夾,以及處理C需要單獨處理的其他許多事情,每個項目和團隊以不同的方式處理它們。
然而,在Rust中被吹捧為優勢的東西對於C開發者來說可能並沒有太大吸引力。Rust的編譯時安全功能無法禁用,因此即使是最小的Rust程序也必須符合Rust的內存安全限制。默認情況下,C可能不太安全,但在必要時它更靈活,更寬容。
另一個可能的缺點是Rust語言的大小。即使考慮到標準庫,C的功能也相對較少。Rust功能集非常龐大並且還在不斷增長。與C ++一樣,較大的Rust功能集意味著更強大的功能,但也意味著更高的複雜度。C是一種較小的語言,但更容易在頭腦中進行建模,因此可能更適合那些對Rust來說太小,不值得大動干戈的項目。
C vs. Python
當今,每當談論軟件開發時,Python似乎總是能出現在對話中。畢竟,Python是“對所有項目的第二佳語言”,毫無疑問是最通用的語言之一,擁有數千個第三方庫。
Python強調的,以及它與C最不同的地方,是有利於開發速度而不是執行速度。一個可能需要一個小時才能用另一種語言寫出來的程序——比如C——可能用Python幾分鐘內就能寫好。另一方面,該程序在C中執行可能需要幾秒鐘,但需要一分鐘才能在Python中運行完。(一個很好的經驗法則:Python程序通常比它們對應的C語言程序運行速度慢一個數量級。)但是對於現代硬件上的許多工作,Python已經足夠快了,這是它如今廣泛應用的一個重要原因。
另一個主要區別是內存管理。Python程序完全由Python運行時進行內存管理,因此開發人員不必擔心分配和釋放內存的細節。但同樣,開發人員的輕鬆也是以運行時性能為代價的。編寫C程序需要嚴格關注內存管理,但生成的程序通常是純機器速度的黃金標準。
但是,在二者的血脈中,Python和C共享一個深層的關係:Python運行時參考是用C語言編寫的。這允許Python程序包裝用C和C ++編寫的庫。第三方庫的Python生態系統的很多重要模塊,例如機器學習方面的庫,其核心是C代碼。
如果開發速度比執行速度更重要,並且如果程序的大多數高性能部分可以被隔離到獨立組件中(而不是遍布整個代碼),那麼純Python或Python和C庫的混合使得會是比單獨使用C更好的選擇。否則的話,C仍然是老大。
03、C語言為何值得去學
1.嵌入式領域,C語言依然是首選語言,嵌入式並沒有因為其他上層語言的發展而沒落,現在嵌入式依然還在其自身的領域展現強大的生命力。手機,電視機,機頂盒,空氣淨化器等等電子產品都是其領域範疇,從長遠看短時間內不可能被消失。而且智能機器人的崛起,C語言的使用頻率又開始加大了。
2.操作系統內核代碼還是C語言為主打,就語言的靈活性以及執行的效率來看C語言還是最合適的語言,而且在系統層次的代碼,C語言還是首選語言。而且現在很多流行語言的底層絕大部分的C語言構建完成。從這個層面講C語言是永遠不會過時的,頂多算是應用範圍變窄,但其作用依然強大。
3. C語言的職位比例相對應用級語言是低了點,但是整個軟件行業在發展,絕對的C語言編程職位並沒有減少。而且對於有志於成為架構師層次的程序員來說,C語言還是必修課,構建軟件框架還是需要對底層有所了解。