摘要:C语言在这过去的五十年间,已经逐渐发展成为极其重要的软件开发语言。本文将深度剖析它是如何与C++、Java、C#、Go、Rust和Python进行竞争,并保持一定优势的。
对于计算机编程语言来说,没有什么技术能沿用半个世纪年,除非它比其他的都好用。C语言于上世纪七十年代初年面世,如今在软件世界仍保持着底层基础开发的主流语言的地位。
有时,一个技术能被长久地流传使用,是因为人们还没有找到一个更好的替代品。在过去的几十年,涌现了大量的语言——甚至出现专门为了挑战C语言的统治地位而设计的新语言。
C真的很难被替代。编程语言调查和软件开发实例都印证了可以用远比使用C语言更好的方式来做开发。但C的地位仍岿然不动,因为它的背后是几十年的积淀和进步。几乎没有语言可以在性能、逻辑、或者普遍性上打败它。
C vs. C++
很自然地,人们会拿C与C++做对比,顾名思义,C++是从C发展而来的。两者之间的不同就在于易扩展性,或者易用性。
语法和方式上,C++与C语言比较接近,但C++提供了很多原生C没有的有用特性:命名空间、模板、异常、内存管理。项目如果对于性能比较敏感,例如数据库和机器学习,通常使用C++编写会对提供系统提高性能更有帮助。
除此之外,C++比C更容易扩展。C++20甚至带来更多的新特性,包括模块、协程、同步库以及其他概念,这些都让模板更易使用。对标准C的最新修订几乎没有新增特性,而是更注重保持向后兼容性。
事实上,所有C++的优点也是它的缺点。C++的特性使用得越多就越复杂,结果就越加难以控制。所以,把自己限制于C++子集的开发者,能避免很多严重的问题。也有些机构想避免C++复杂性,坚持使用C。比如Linux内核的开发团队就会避开C++。
对于前期开发者和维护代码人员来说,选择C是一个避免C++过度使用纠纷的方式。不过,C++也有一系列丰富的高级功能,该用还是要用。但如果简洁明了更适合现在或者未来项目的整体发展的话,C会更有优势。
C vs. Java
在JAVA诞生了几十年之后,其仍然是一般企业级软件开发的主要语言。大多数优秀的企业软件开发项目都是用Java编写,包括绝大多数Apache基金会项目。当开发企业级项目时,Java是一个可行性比较高的语言。
Java的语法大量地借鉴了C和C++。不过与C不同的是,Java不会默认编译成机器语言。相反地,Java运行时环境JVM会将Java代码即时编译到目标环境中运行。在良好的条件下,即时编译的Java可以达到甚至超C的性能。
Java奉行的“一次编写,随处运行”的思想,Java程序仅需小的调整,就能运行在不同的环境。相比之下,尽管C已经移植到许多体系结构中,但是任何给定的C程序可能仍然需要定制才能在Windows和Linux上正常运行。
其可移植性和强大性能的结合,以及软件库和框架的庞大生态系统,使Java成为企业级项目语言的一员。
然而,Java落后C的地方在于,Java从来没有竞争的领域,都是接近底层运行,或直接操作硬件。C代码被转换成机器码,由进程直接执行。Java被编译成字节码,由JVM解释器转换为机器代码的中间代码。此外,尽管Java的自动内存管理在大多数情况下是一种好事,但是C更适合于对有限内存资源有优化要求的程序。
在某些情况下,Java的性能可以接近C。JVM的JIT引擎能在运行时根据程序的行为优化程序,可以进行许多种类的优化,对于预先编辑的C语言而言,这个是行不通的。例如,ApacheSpark使用自定义的内存管理代码绕过JVM进行了一定程度的内存内处理优化。
C vs. C# 与 .Net
在推出近20年之后,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编译允许对正在运行的.NET程序进行各种优化,这在C中是无法进行的。
和C一样,C和.NET提供了各种直接访问内存的机制。堆、堆栈和非托管系统内存都可以通过.NETapi和对象进行访问。开发人员可以使用.NET中的unsafe模式来实现更高的性能。
不过,值得注意的是,托管对象和unsafe对象之间不能随意交换,它们之间的封装传送需要以降低性能作为代价。因此,减少两者之间的传递,可以最大化的提高.NET程序的性能。
当负担不起托管内存相对于非托管内存的代价时,或者当.NET运行时对于目标环境(如内核空间)是一个很糟糕的选择项或者根本不可用时,那么C语言或许就能解决问题了。与C和.NET不同,C默认情况下会开启直接内存访问。
C vs. Go
Go语法和C很像,大括号作为分隔符、以分号结尾语句。精通C的开发人员通常无需太多困难就可以直接转入Go,甚至把Go的新特性如名称空间和包管理考虑在内也是如此。
代码的易读性是Go的指导设计目标之一,开发人员能够轻松地跟上任何Go项目的速度,并在短时间内精通代码库。C代码库很难摸索,因为它们很容易变成一个由宏和特定于项目或团队的嵌套。Go的语法,以及其内置的代码格式和项目管理工具,都是为了避免这些机制问题。
Go还提供额外的功能,像Goroutines和Channels,用于处理并发性的语言级工具以及组件之间的消息传递。在C语言里面只能自己实现或者用三方库,但是Go以开箱即用的方式提供了这些特性,让我们在开发需要类似功能的软件的时候,变得极其方便。
在内存管理方面,Go与C有较大区别。默认情况下,Go对象被自动管理和回收。对于大多数编程工作来说,这非常方便。但这也意味着任何需要对内存进行特殊处理的程序,会比较难办。
Go的确包含了一个unsafe的包,用于规避Go的一些类型处理的安全性问题,例如使用Pointer类型读取和写入任意内存。但unsafe伴有一个警告,即用它编写的程序“可能不可移植,并且不受Go1兼容性准则保护”。
Go非常适合构建命令行程序和网络服务等程序,因为它们很少需要这样的细粒度操作。但是低级的设备驱动、内核空间操作系统组件以及其他需要对内存布局和管理进行严格控制的任务最好是在C中创建。
C vs Rust
在某些方面,Rust是解决C和C++造成的内存管理难题的新方案,也是解决这些语言许多其他缺点的新方案。Rust编译为本机代码,因此在性能上与C相当。不过,默认情况下,内存安全是Rust的主要卖点。
Rust的语法和编译规则能帮助开发者避免常见的内存管理错误。如果一个程序存在跨过Rust语法的内存管理问题,那么它就不会编译。使用该语言的新手,尤其是从像C这样为此类错误提供了大量空间的语言转过来的新手,他们学习Rust的第一阶段是如何安抚编译器。但是Rust支持者认为,这种短期的痛苦将得到一个长期的回报:不会牺牲速度的更安全的代码。
Rust也可以用它的工具改善C。默认情况下,项目和组件管理是Rust提供的工具链的一部分,与Go相同。有一种默认的、推荐的方式来管理包、组织项目文件夹,以及处理许多其他事情,这最多是临时措施,每个项目和团队处理它们的方式都是不同的。
尽管如此,对于C开发人员来说,被吹捧为Rust优势的东西可能看起来不是那样的。Rust的编译时安全特性不能被禁用,所以即使是再小的Rust程序也必须符合Rust的内存安全限制。默认情况下,C可能不太安全,但在必要时,它更灵活,更宽容。
另一个可能的缺点是Rust语言的体积。即使考虑到标准库,C的新特性也相对较少。Rust特性集正在蔓生并持续增长。与C++相比,较大的Rust特性集意味着更强大的能力,但也更复杂。C是一种较小的语言,但更容易建模,因此可能更适合于看上去有点臃肿的项目中。
C vs Python
现在,每当谈论软件开发时,Python似乎总是会被人们提起。毕竟这是“第二个适合所有事情的语言”,毫无疑问,它是最通用的语言之一,有数千个第三方库。
Python强调的是开发速度而不是执行速度,这是它与C的最大区别。用C语言组装一个程序可能需要一个小时,而用Python只需几分钟。另一方面,该程序在C语言中执行可能只需要几秒钟,而在Python中运行则需要一分钟。但是对于现代硬件来说,Python足够快,这是它获得成功的关键。
另一个主要区别在于内存管理。Python程序完全是由Python运行时进行内存管理,因此开发人员不必担心分配和释放内存的困难。但这里有必要讲明,开发者的轻松是以牺牲运行时性能为代价的,编写C程序需要谨慎地注意内存管理。
其实,Python和C之间有一个很深的联系:参考Python运行时是用C写的。这允许Python程序打包C和C++编写的库。Python生态系统中一些重要的第三方库,如机器学习,其核心是C代码。
如果开发速度比执行速度更重要,程序执行部分可以隔离成独立的组件,而不是分散在整个代码中,那么纯Python或Python和C库的混合比单独使用C更好。否则的话,C仍然是霸主。
感谢XJRsoft提供撰文支持,原文来自Serdar Yegulalp
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。