写点什么

稳坐开发领域霸主之位,揭秘 C 语言无可取代的几大原因!

发布于: 2020 年 12 月 03 日

摘要: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

详情参阅:https://www.xjrsoft.com

 

点击关注,第一时间了解华为云新鲜技术~


发布于: 2020 年 12 月 03 日阅读数: 39
用户头像

提供全面深入的云计算技术干货 2020.07.14 加入

华为云开发者社区,提供全面深入的云计算前景分析、丰富的技术干货、程序样例,分享华为云前沿资讯动态,方便开发者快速成长与发展,欢迎提问、互动,多方位了解云计算! 传送门:https://bbs.huaweicloud.com/

评论

发布
暂无评论
稳坐开发领域霸主之位,揭秘C语言无可取代的几大原因!