[欢度国庆]为什么我们今天还要学习和使用C++?(转载)

在各种新的开发语言层出不穷的今天,在Java和C#大行其道今天,我们为什么还要学习和使用C++?现在学习C++将来有用吗?学习C++要花费那么多时间和精力,这一切都值得吗?现在学习C++有钱途吗?
这几乎是每一个C++的初学者都会问的问题,在选择开发语言进行学习的时候,就像高考填报志愿一样,都会有些迷茫。下面这篇Herb Sutter的文章(其实是他在 C++ and Beyond 2011上的一次公开演讲)很好地回答了这些问题。所有那些正在学习C++而又对C++的前途感到迷茫的同学,都值得一读。
本文系转载,感谢原作者刘未鹏,感谢翻译者waterwall。更重要的,感谢Herb Sutter。
 在这里,还有一篇关于这一演讲的讨论,也可以参考:http://www.cnbeta.com/articles/171221.htm
 

为什么C++?

刘未鹏(pongba) /文

waterwalk /译

C++的罗浮宫(http://blog.csdn.net/pongba)

首先非常感谢waterwalk的辛勤翻译:-) waterwalk把翻译回贴在原文的下面了,为了方便阅读我提取出来编辑以后重发一个帖子。这篇文章原本是想对最近C/C++争论系统的整理一下一些思考的,但由于一开始的时候用英文写了两段,后来就干脆都用英文了,造成很多人阅读的麻烦,在此抱歉。不过好在waterwalk翻译了整篇文章,于是单独贴在这里:-)

另,原文在这里

问题

为什么用C++呢? 在你皱着眉头离开之前,试着回答这个简单的问题。效率,是么?人人都知道这个。但情况是,当一个人开始讨论编程语言或与其相关的话题时,他必须要非常明确而有针对性。为什么呢?我来问你另一个问题:如果效率是人们使用C++的唯一理由,那么为啥不直接用C呢?C被认为比C++效率更高(嗯嗯,我知道C没有比C++的效率高多少,所以这里别误解我的意思,因为即使它们二者效率相同,刚才的问题依然存在)。

迷思

我知道你又要说“更好的抽象机制”了,因为毕竟C++是要设计成一个更好的C的。C++没有牺牲效率,同时又添加了这么多高级特性。但问题是,“开发者们真的需要这些高级特性么?”。毕竟我们一直听人讲KISS之类的东西。我们也都听到有声称C比C++更KISS所以我们要用C云云。这种持续不断的争论将CC++之间的比较变成了一个大大的迷题(或者说是混乱)。令人惊讶的是,貌似的确有很多人更加倾向于用C,最大的理由就是C++实在是太难用对了甚至Linus也这么想

这种现象最大的影响就是当人们在C和C++之间权衡时,使人们倾向于使用C。而且一旦人们开始用C,他们很快就适应并满足了(其实,在任何语言乃至任何人类活动中都有此现象,C++亦然,比如常常听到有人说“XX语言我用了这么多年,一直用得好好的”,照这种说法任何图灵完备的语言还不都是能用来编程?)。于是即使他们还没有试试C++,或者他们还没成为好的C++程序员时,他们就开始声称C比C++更好了。然而其实呢,真实的答案往往总是取决于实际情况的

我说过“取决于实际情况”了么?那到底实际情况是什么呢?显然,有些领域C是更好的选择。例如设备驱动开发就不需要那些OOP/GP技巧。而只是简单的处理数据,真正重要的是程序员确切地知道系统是如何运转的,以及他们正在做什么。那么写操作系统呢?我本人并没有参与任何操作系统的开发,但我读过不少操作系统代码(大多是unix的)。我的感觉是操作系统很大一部分也不需要OOP/GP。

但是,这就表示在所有效率重要的领域,C都是比C++更好的选择么?未必。

答案

让我们一个一个来分析。

首先,当人们关注效率时,有2种效率——时间效率(例如OS,运行时库,实时应用程序,high-demanding的系统)和空间效率(例如各种嵌入式系统)。但是,这样的分类并不能帮我们决定用C还是C++,因为C和C++的时空效率都很高。真正影响选择语言的因素是业务逻辑(这里的“业务逻辑”并非表示“企业应用业务”)。例如,使用OOP/GP来表达逻辑(或者说代码的结构)好呢,还是就只用数据和过程好呢?

据此观点,我们可以把应用程序大致分为两类(当然前提是关注的是C/C++而不是java/C#/ruby/erlang等等):底层应用程序和高层应用程序。这里底层是指像OB/OO和GP没啥用处的地方, 其余归到高层。显然,在所有C/C++应用的领域(这些领域需要C/C++的效率),属于高层的应用有很多(可以看看Bjarne Stroustrup在他主页上的列表)。在这些领域中,抽象至少是和效率一样重要的。而这些正是C++适用的场合。

等等还有。即使在程序员不需要高级抽象的领域,也不是就绝对用不到C++。为啥呢?仅仅是因为你的代码中没有用类或模板并不意味着不能用以类或模板实现的库。因为有如此众多方便的C++库(还有即将到来的tr1/tr2),我觉得有充分的理由在这些领域中使用C++——你可以在编码时仅使用C++中的C核心(以任何你喜欢的方式来KISS),同时还能用强大的C++(比如STL容器、算法和tr1/tr2的组件)。

最后,我认为人们还常常忽略了一点——有时KISS是建立在抽象上的。我觉得Matthew Wilson在他新书《Extended STL,卷1》的序言中对此做了很好的阐释。他写了2段代码,一段用C,另一段用C++:

// in C
DIR*  dir = opendir(".");
if(NULL != dir)
{
  struct dirent*  de;
  for(; NULL != (de = readdir(dir)); )
  {
    struct stat st;
    if( 0 == stat(de->d_name, &st) &&
        S_IFREG == (st.st_mode & S_IFMT))
    {
         remove(de->d_name);
    }
  }

  closedir(dir);
}

// in C++
readdir_sequence entries(".", readdir_sequence::files);
std::for_each(entries.begin(), entries.end(), ::remove);

//而在C++09里面更简单:
// in C++09
std::for_each(readdir_sequence(".", readdir_sequence::files), ::remove);

也就是说,我认为即使一个人在自己的代码里不需要类或模版,他也有理由用C++,因为他用的那些方便的C++库用到了类和模板。如果一个高效的容器(或智能指针)能把你从无聊的手动内存管理中解放出来,为啥还要用那原始的malloc/free呢?如果一个更好的string类(我可没说std::string,地球人都知道那个不是C++中能做出的最好的string类)或正则表达式类能把你从一坨一坨的、你看都不想看的处理字符串的代码中解脱出来,那么为啥还要手动去做这些事呢?如果一个 "transform"(或"for_each")能够用一行代码把事情漂亮搞定,为啥还要手写一个for循环呢?如果高阶函数能满足你的需要,那么为啥还要用笨拙的替代方法呢?(OK,我知道,最后两个需要C++加入lambda支持才真正摆脱鸡肋的骂名——这正是C++0x的任务嘛)

总之,我认为KISS并不等同于原始KISS意味着用最适合的工具来做事情,这里最合适的意思是工具能够帮你以尽量直接简洁的方式来表达思想,同时又不降低代码的可读性,另外还保持代码容易理解。

真正的问题

人们可能会说,相较于被正确使用而言,C++(远远)更容易被错误使用。而相比而言,C程序的复杂性更容易管理和控制。在C++中,一个普通程序员很可能会写出一堆高度耦合的类,很快情况就变得一团糟。但这个其实是另外一个问题。在另一方面,这种事情也很可能发生在任何一门面向对象语言中,因为总是有程序员在还没弄懂什么是HAS-AIS-A前,就敢于在类上再写类,叠床架屋的一层一层摞上去。他们学会了在一门特定的语言中如何定义类,如何继承类的语法,然后他们就认为自己已经掌握了OOP的精髓了。另一方面,这一问题在C++中更为严重,因为C++有如此众多的偶然复杂性在阻碍设计;而且C++又是如此灵活,很多问题在C++中都有好几种解决办法(想想那么多的GUI库吧),于是在这些选择中进行权衡本身就成了一个困难。C++中的非本质复杂性是其历史包袱使然,而C++0x正是要努力消除这些非本质复杂性(在这方面C++0x的工作的确做得很不错)。对于设计来说,灵活性不是个坏事情——可以帮助好的设计者作出好的设计。如果有人抱怨说这个太费脑细胞了,那可能是这个设计者本身的问题,而不能怪语言。可能就不该让他来作设计。如果你担心C++的高级特性会把你的同事引入歧途,把项目搞砸,那你也许应该制定一份编码标准并严格推行(或者你也可以遵循C++社群这些年积攒下来的智慧,或者在必要时,只使用C++中的CC with class那部分),而不是因为有风险就躲开C++(其实这些风险可以通过一些政策来避免的),因为那样的话,你就没法用那些C++的库了

另一方面,其实一个更为重要的问题是一个心理学问题——如果一门语言中存在某个奇异的特性或旮旯,那么迟早总会有人发现的,总会有人为之吸引的,然后就使人们从真正有用的事情中分心出来(这有点像Murphy法则),更不用说那些有可能对真正问题带来(在某种程度上)漂亮的解决方案的语言旮旯了。人们本性上就容易受到稀有资源的诱惑。奇技淫巧是稀有资源,于是奇技淫巧便容易吸引人们的注意力,更别说掌握一个技巧还能够让那人在他那圈子里感觉非常牛了。退一万步,你会发现,即使是一个废柴技巧也能引起人们足够的兴趣来。

C++中有多少阴暗角落呢?C++中又有多少技巧呢?总的来说,C++中,有多少非本质复杂性呢?(懂一定C++的人一定知道我在说什么)

平心而论,近年来(现代C++中)发现的大多数技巧或(如果你愿意称之为)技术实际上都是由实际需求驱动的,尤其是需要实现高度灵活而又普遍适用(generic)的类库 (例如boost中的那些玩意)。而这些技巧也的确(在某种程度上)提供了对实际问题的漂亮解决方案。让我们来这么想一下,如果你处于一个两难境地:要么用那些奇技淫巧来做点很有用的东西,要么不做这样其他人也就没得用。你会如何选择呢?我知道boost的英雄们选择了前者——不管多么困难多么变态多么龌龊,把它做出来!

但所有这些争论都不能改变一个事实我们理应享有一个语言,能够让我们用代码清晰的表达思想。以boost.function/boost.bind/boost.tuple为例,variadic templates可以大大简化这几个库的实现(减至几乎是原先1/10的代码行数),同时代码也(远远)更加简洁易懂。Auto,initializer-list,rvalue-reference,template-aliasing,strong-typed enums,delegating-constructors,constexpr,alignments,inheriting-constructors,等等等等,所有这些C++0x的特性,都有一个共同目的——消除语言中多方面的非本质复杂性或语言中的尴尬之处

正如Bjarne Stroustrup所说,很显然C++太过复杂了,很显然人们被吓坏了,并且时不时就不用C++了。但“人们需要相对复杂的语言去解决绝对复杂的问”。我们不能通过减少语言特性而使其更加强大。复杂的特性就连模板甚至多继承这样的也是有用的——如果你正好需要它们,而且如果你极其小心使用,不要搬起石头砸自己的脚的话。其实在所有C++的复杂性当中,真正阻碍了我们的是非本质复杂性(有人称之为“尴尬之处”),而不是语言所支持的编程范式(其实也就3个而已)。而这也正是我们应该拥抱C++0x的重要原因,因为C++0x正是要消除那些长期存在的非本质复杂性,同时也使得那些奇技淫巧不再必要(很显然,目前这些技巧堆积如山,翻翻那些个C++的书籍,或者瞅瞅boost库,你就知道我在说啥了),这样我们就能够直观清晰的表达思想

结论

C++难用,更难用对。所以当你决定用它时,要小心,要时刻牢记自己的需求所要达到的目的。这里有一个简单的指南:

我们需要高效率么?

如果需要,那么

我们需要抽象么(请仔细思考这一点,因为很难评估使用C++高级特性是否能够抵消误用这些机制的风险正确的回答取决于程序员的水平有多高,遵循哪种编码标准以及编码标准执行得如何,等等)?

如果是,那么用C++吧。如果不是,那么,

我们需要用C++库来简化开发么

如果是,那就用C++吧。但同时必须时刻牢记你在做什么——如果你的代码不需要那些“漂亮的”抽象,那就别试图使用以免陷入其中。别只是因为你在.cpp文件中写代码以及你用的是C++编译器就要用类啊、模板啊这些东西

如果不是,那就用C,不过你又会想为啥不仅仅使用C++中属于C的那部分核心呢?还是老原因:人们很容易就陷入到语言的“漂亮”特性中去了,即使他们还不知道这些特性是否有用我都记不清有多少次自己写了一大堆的类和继承,到最后反倒要问自己要这么些个类和继承做什么呀?。所以,如果你能坚持只用C++CC with class的那部分并遵循“让简单的事情保持简单”的理念;或者你需要把C代码迁移到C++中来的话,那么就用C++吧,但要十分小心。另一方面,如果你既不需要抽象机制,也不需要C++库,因为事情非常简单,不需要方便的组件例如容器和字符串,或者你已认定C++能够给项目带来的好处微乎其微,不值得为之冒风险,或者干脆就没那么多人能用好C++,那么可能你还是只用C的好

底线是:让简单的事情保持简单(但同时也请记住:简单性可以通过使用高级库来获得);必要时才使用抽象(切记不可滥用;遵循好的设计方法和最佳实践)。

时间: 2024-11-05 09:26:33

[欢度国庆]为什么我们今天还要学习和使用C++?(转载)的相关文章

AspectJ基础学习之三HelloWorld(转载)

AspectJ基础学习之三HelloWorld(转载) 一.创建项目 我们将project命名为:aspectjDemo.然后我们新建2个package:com.aspectj.demo.aspect 和 com.aspectj.demo.test 前者用来方apsect.后者用来放测试类.如果你仔细的话,你会发现Aspectj的项目上面有个AJ的标志. 二.创建Aspect 首先我们创建HelloWorld.java.他包含main()方法,但是没有方法体,代码如下: [java] view

JavaWeb学习总结——Session(转载)

JavaWeb学习总结--Session(转载) 一.Session简单介绍 在WEB开发中,服务器可以为每个用户浏览器创建一个会话对象(session对象),注意:一个浏览器独占一个session对象(默认情况下).因此,在需要保存用户数据时,服务器程序可以把用户数据写到用户浏览器独占的session中,当用户使用浏览器访问其它程序时,其它程序可以从用户的session中取出该用户的数据,为用户服务. 二.Session和Cookie的主要区别 Cookie是把用户的数据写给用户的浏览器. S

Android学习之路(转载)

原文地址:http://stormzhang.github.io/android/2014/07/07/learn-android-from-rookie/ 硬件 电脑–推荐Mac 首先声明我不是果粉,个人Windows,Linux,Mac OX系统均用过, 只能说Windows上面的开发工具简直难以恭维,尤其命令行超级难用,而Linux自己必须得花不少时间在折腾中,更是不适合新手了,Max OS是我认为迄今为止最好用的系统,没有之一, 所以如果你不差钱的话,强烈建议入手一台Mac,推荐Pro系

向“生物力学之父”冯元桢先生学习什么?【转载】

[转载] 原帖:http://blog.sciencenet.cn/blog-485553-753738.html 在企业从事技术工作的我本该与“生物力学之父”冯元桢先生没什么交集,但因本人从事机械行业的流体传动与控制技术领域,幻想能否用流体力学与机械结构作类比,解析高血压现象,于是找来一本<人体解剖学>的书籍来看(仅看了血液循环一章),发现血液循环系统和流体传动系统原理是一样的,心脏.瓣膜.血管.血液完全可用泵.阀.管道.矿物油作类比.流体传动系统要服从流体力学,机构学,热力学规律,血液循环

毕竟是国庆假期,并没有学习什么。

10月5号了,这个国庆假期准备过去. 到了学校又要开始学习和做好网站的管理了. 假期假期,我真的只是想好好的休息休息. 有的时候有的事情真的很累.我也不想矫情,只想开开心心简简单单地生活. 今天朋友叫去KTV,然后发了KTV的照片给我.我一眼就看到我的前任在里面,就用我很忙拒绝了聚会. 我很忙真的是一个很好的借口.我不懂,也不知道.是不是我说我不想见她就是一种罪过,就是我内心放不下. 我只是单纯不想见,不想去见了之后又犯矫情.错过了就错过了,纠缠那么多干什么. 好好珍惜眼前的生活. 今天在姐姐的

为什么还要学习TypeScript

什么是TypeScript TypeScript 是 JavaScript 的一个超集,主要提供了类型系统和对 ES6 的支持,它由 Microsoft 开发,代码开源于 GitHub 上. 它的第一个版本发布于 2012 年 10 月,经历了多次更新后,现在已成为前端社区中不可忽视的力量,不仅在 Microsoft 内部得到广泛运用,而且 Google 的 Angular2 也使用了 TypeScript 作为开发语言. 为什么学习 TypeScript TypeScript 是由 C#语言的

我的算法学习之路(转载)

关于 严格来说,本文题目应该是我的数据结构和算法学习之路,但这个写法实在太绕口——况且CS中的算法往往暗指数据结构和算法(例如算法导论指的实际上是数据结构和算法导论),所以我认为本文题目是合理的. 如果你使用的是手机或平板设备,那么请点击下面的链接以获得更好的阅读效果: http://zh.lucida.me/blog/on-learning-algorithms/ 原文作者:Lucida 这篇文章讲了什么? 我这些年学习数据结构和算法的总结. 一些不错的算法书籍和教程. 算法的重要性. 初学

怎样学习一门新技术(转载)

几周前一个朋友也打算通过跑步来甩一甩脂肪,控制一下自己那已经严重扭曲了的身材 —— 他便求教我从衣服到鞋,从腕表到心率带,都买了什么装备,性能如何,他好照着原样来一套.我告诉他我什么装备都没买,跑了半年多了,我还是随便套上件纯棉的T恤,穿条舒服的运动短裤,蹬着跑鞋,抓着手机,就这么跑将起来.他连连说不信,一个连排汗速干的衣服,臂包,运动腕表都没配的人,怎么也能(配得上)跑步? 「我想学rails,有没有推荐的ruby的入门书籍?」 「我想学android,对java掌握到什么程度才能开始呢?」

OpenFOAM 学习路线 【转载】

“Two weeks of playing with a CFD code will save you one afternoon of reading” 什么是OpenFOAM(力荐) 注意,下面列的资料都在不断更新,本站博文也不断更新. OpenFOAM国内使用情况调查 在大家眼里OpenFOAM难点在哪? 不断有新人加入OpenFOAM的学习和使用中来,因此一些问题反复被提起,比如:刚开始怎么学openfoam? 因此这里花一点时间稍微总结一些入门的建议,首先通读本文可以事半功倍. 先学操