概述:学习和使用过C++的人几乎都曾经听说过下面的五个关于C++的观点,并且对这些话笃信不已,那么真实的情况是怎么样的呢?本文的作者——C++之父Bjarne Stroustrup将会对这些观点作逐一回击。
以下的这五个观点盛行于C++多年:
- “要了解C++,你必须先学习C语言。”
- “C++是一门面向对象的语言。”
- “对于可靠的软件,垃圾回收机制必不可少。”
- “为了提高效率,你必须编写底层代码。”
- “C++只对大型复杂的项目有用。”
如果你还对这些观点深信不已,那么这篇文章可以给你一些重新认识。这些观点在特定的时间对于某些人、某些工作来说是正确的。但是对于今天的C++,随着ISO C++11标准的编译器和工具的广泛使用,这些观点都需要被重新认识。
- C++之父谈关于C++的五个需要被重新认识的观点(上)
- C++之父谈关于C++的五个需要被重新认识的观点(中)
接下来,我们将会对这些观点进行逐一反驳。
观点一:“要了解C++,你必须先学习C语言。”
这不对,事实上对于基础编程的学习来说C++比C语言容易的多。
C语言虽然几乎可以认为是C++的子集,但对初学者来说却不是最容易学习的。因为C语言缺乏标记支持和类型安全,并且对于简化简单任务来说C++的标准库更加易于使用。
比如,对于一个非常简单的用于描述邮件地址格式的函数:
它可以被这样使用:
而C语言中需要明确的字符操作和明确的内存管理:
然后,它需要被这样使用:
相比之下,哪种版本更加容易学习?哪种语言更加有效率?很显然是C++了,因为它不需要计算参数字符,不需要为简短的字符串分配动态内存。
关于“C语言优先学习”的观点并非来自少部分人的认识。传授这种典型观点的老师主要有以下几个方面原因:
- 因为这是他们在这方面有丰富经验。
- 因为这是课程需求。
- 因为这是老师年轻时的学习方式。
- 因为C比C++要小,所以更容易学习。
- 因为学生迟早都必须学习C语言或者C++的C语言子集。
然而,C语言并不是作为优先学习的最简单和有用的C++子集。当你知道足够多的C++知识后学习C语言则会非常容易。这种学习方式可以有效减轻从C到C++学习时在认识和技术上的困难。
对于现代C++的教学方法,可以参见我的著作:Programming: Principles and Practice Using C++。它甚至在有一章的结尾处展示了如何学习使用C语言。这种教学方法在几所大学的数以万计学生中使用,非常的成功。它的第二版是使用C++11和 C++14来让学习变得更加容易。
C++11标准使C++更容易被初学者接受,例如,这里是一个元素序列已初始化的vector标准库:
在C++98中,我们只能初始化数组和列表。在C++11中,我们可以定义一个包含有{}和需要的任何类型的初始化列表的构造函数。
我们可以通过for循环的范围来遍历vector:
对于v的任何一个元素都会调用一次test()。
for循环的范围可以遍历任何序列,因此我们可以通过直接使用初始化列表来简化示例。
C++11的目的是使简单的事情变得简单。代码的简单化并没有以性能降低为代价。
观点二:“C++是一门面向对象的语言。”
不对。C++支持面向对象和其它编程风格,它并不仅限于“面向对象”这个狭隘的观点。它支持一个综合的编程技术,包括面向对象和泛型编程。通常一个问题的最佳方式需要比较多种类型。最佳,在这里指的是时间最短、最易于理解、最有效率和最易于维护等等。
“C++是一门面向对象的语言”的观点使人们在除非需要拥有许多虚拟(多态运行)函数的巨大类层次结构时才会考虑使用它。而这种用法对于许多问题来说是不合适的。这个观点也会导致另外一些人指责C++的面向对象并不纯粹。毕竟,如果把“好”和“面向对象”划上等号的话,C++还包含了其它被认为是“不好”的非面向对象的东西。这种观点产生的两种认识都会导致人们放弃学习C++。(译者注:作者表达的意思就是把C++比作是一个卖包子和卖米线的餐馆。将C++认作是包子铺会让人产生2种误会,其一,路过的人会以为这里只卖包子,不卖其它的;其二,爱吃包子的人会认为包子铺还卖米线,这包子一定做得不专业)
举个例子:
它面向对象吗?当然,它严重依赖包含虚函数的类层次结构。它是泛型编程吗?当然是,它严重依赖于参数化容器(vector)和泛型函数for_each。它是函数式编程吗?在一定程度上是,它使用了匿名函数(由[]构造)。那么它到底是什么?它是现代C++:C++11。
我同时使用了for循环和标准库算法for_each只是为了展示其特性。在实际代码中,我只会使用其中的一个循环。
你想让上面那段代码更通用吗?因为毕竟它只适用于vector指针的Shape基类。那么对于列表和内置数组呢?对于象shared_ptr和unique_ptr这样的“智能指针”(资源管理指针)呢?对于没有调用Shape类的对象能够使用draw()和rotate()么?可以这样来做:
你可以使用这段程序对任何序列从头到尾进行遍历。这是一个C++风格的标准库算法。我使用了auto来避免必须为“象Shape类这样的对象”的接口类型命名。这是C++11的特性,它的含义是“使用被用于初始化的表达式的类型”。所以由for循环中p的类型就能决定这是什么类型的对象。这种使用auto表示匿名函数参数类型的方法是现已广泛使用的一个C++14新特性。
如下图所示:
在这里我假定Blob是包含了操作函数draw()和rotate()的图形化类型,而Container是容器类型。标准库list(std::list)拥有成员函数begin()和end(),用于帮助用户遍历元素的序列。这是很好很经典的面向对象编程。但是,假如容器不支持C++标准关于遍历半开序列[b:e)的概念呢?假如库里面没有begin()和end()成员函数呢?或者,由于没有容器一类的东西因此无法遍历。对于这些情况,我们可以用适当的语义来定义独立的begin()和end()。标准库提供了C语言风格的数组,因此如果容器是C语言风格的数组,问题就迎刃而解了——而C语言风格的数组非常常见。
来看看一个更难点的例子,假如容器保留了对象的指针,并且有一个用于访问和遍历的不同模型呢?比如,你会访问到象下面的这个容器:
这种风格并不少见,我们可以将其映射到[b,e)这样的一个序列:
注意,这种修改是无关紧要的:我并没有修改容器或者某些由C++标准库支持的将容器映射到模型进行遍历的容器类的层次结构。这是改写的一种形式而不是重构。
我选择这个例子是为了说明这些泛型编程技术并不局限于流行的标准库。它们也符合常见的“面向对象”的定义,但是它们却不是面向对象的。
关于C++的代码一定是面向对象(意味着在每个地方都会使用层次结构和虚函数)的观点深深地影响了人们对C++性能的评价。还有一些人认为当需要解决多种类型的运行的问题只有面向对象才是最好的。在以前,我也是这么想的。但是事实上,它也有死板的一面(比如并不是所有相关类型都属于同一层次结构)并且虚函数无法作为内联函数(这就使得处理许多简单而重要的任务时会多花费大量的时间)。
下一篇将会围绕“对于可靠的软件,垃圾回收机制必不可少。”的观点进行说明……
本文翻译自Five Popular Myths about C++, Part 1,作者为:C++之父Bjarne Stroustrup
相关推荐
C++程序设计语言(特别版) 习题详解(含目录) c++ 之父之作 将晦涩的c++演绎。。。。。
本书由C++语言 的设计者编写的,是有关C++语言最全面、最权威的著作。本书涵盖了标准C++以及由C++所支持的关键性技术和设计技术。标准C++较以前的版本功能更强大,其中许多新的语言特性,如名字空间、异常、米板、...
书中几乎介绍了C++语言的全部核心功能和重要的标准库组件,以很短的篇幅将C++语言的主要特性呈现给读者,并给出一些关键示例,让读者用很短的时间就能对现代C++的概貌有一个清晰的了解,尤其是关于面向对象编程和...
The.Design.And.Evolution.Of.C+是一本关于C++语言设计原理、设计决策和设计哲学的专著。它清晰地回答了C++为什么会成为今天这个样子而没有变成另外一种语言。作为C++语言的创建者,Bjarne淋漓尽致地展示了他独到而...
The C+ + Programming Language(c++英文版 c++之父所著)
21天学通C++(第五版) vs2008程序打包 深入浅出MFC 浙大教材C++达内 C++ 华为C++培训资料 visual c++_MFC 资源内容: visual c++_MFC 达内 C++ C++课件.ppt c++源码 .rar Core C++ Programming_new.ppt Core C++ ...
c/c++ 编写服务程序、后台程序 C/C++ 父进程监控子进程退出后重新执行
C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++C++
要搞C++的,这本绝对要备好,C++老爸写的书,能不要么??顺便补充一下,这个是PDF版本的很好阅读!!
总之,《C++语言程序设计第五版》涵盖了C++的基础到高级知识,结合PPT和代码,为学习者提供了一个全面的学习路径,帮助他们掌握这一强大的编程语言。通过深入学习和实践,可以为进入更复杂的系统开发和算法设计打下...
C++之父Bjarne_Stroustrup对于c++‘注意事项’的经典总结,相信每个无论是c++新手还是老手,都会受益匪浅
《Effective Modern C++:改善C++11和C++14的42个具体做法(影印版)(英文版)》中包括以下主题:剖析花括号初始化、noexcept规范、完美转发、智能指针make函数的优缺点;讲解std∷move,std∷forward,rvalue引用和...
c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏c++小游戏...
首先,从标题和描述中可以得知书籍的全名为《C++大学基础教程(第五版)(C++ How to Program中文版)》,这是一本关于C++编程语言的教材。本书可能是《C++ How to Program》系列的第五版,该系列教材被广泛用于大学...
《C++ Primer》第五版是C++领域内极具权威性的教材,由Stanley B.Lippman, Josée Lajoie, Barbara E.Moo三位专家联合编写。本书覆盖了C++编程语言的各个方面,包括基本概念、程序结构、标准库等,并以示例驱动的...
这个资源包集合了一个多样化的C++编程实现的小游戏,适合对C++编程感兴趣的初学者和爱好者。C++是一种强大的、通用的面向对象编程语言,以其高效性和灵活性闻名,广泛应用于游戏开发、系统软件、嵌入式系统以及各种...
模板是C++的一个高级特性,允许创建泛型代码,即可以在多种数据类型上工作的函数或类。模板可以提高代码的通用性,减少重复工作。 异常处理是处理程序运行时错误的方式,通过try、catch和throw关键字,可以捕获并...
c++之父的A Tour of C++第二版,新增介绍了c++17以及c++20的一些特性