有效的C ++项目23首选非成员非友元函数到成员函数

问题描述 投票:34回答:7

虽然对类设计中的一些事实感到困惑,特别是函数是否应该是成员,但我查看了有效的c ++并找到了第23项,即将非成员非友元函数更喜欢成员函数。第一手阅读Web浏览器示例有一定意义,但是该示例中的便捷函数(在本书中命名为非成员函数)会改变类的状态,不是吗?

  • 所以,第一个问题,他们不应该成为会员吗?
  • 进一步阅读,他认为STL函数,实际上某些类没有实现的函数是在stl中实现的。根据本书的思想,它们演变成一些便利函数,这些函数包含在一些合理的命名空间中,例如来自std::sortstd::copyalgorithm。例如,vector类没有sort函数,并且使用stl sort函数,因此它不是vector类的成员。但是也可以将相同的推理延伸到矢量类中的一些其他函数,例如assign,这样也不能作为成员实现,而是作为便利函数实现。但是,这也会改变对象的内部状态,例如它操作的排序。那么这个微妙但重要(我猜)问题背后的理由是什么呢?

如果你有权访问这本书,你可以为我澄清这些要点吗?

c++ encapsulation member-functions non-member-functions effective-c++
7个回答
36
投票

绝对没有必要访问这本书。

我们在这里讨论的问题是依赖和重用。

在设计良好的软件中,您尝试将项目彼此隔离以减少依赖关系,因为在需要更改时,依赖关系是一个需要克服的障碍。

在精心设计的软件中,您应用DRY原则(不要重复自己),因为当需要进行更改时,必须在十几个不同的地方重复它,这很痛苦,而且容易出错。

“经典”OO思维模式在处理依赖关系方面越来越糟糕。通过直接依赖于类的内部的许多方法,最轻微的改变意味着整个重写。不一定如此。

在C ++中,STL(不是整个标准库)的设计具有以下明确的目标:

  • 削减依赖性
  • 允许重用

因此,容器公开了明确定义的接口,这些接口隐藏了它们的内部表示,但仍然提供了对它们封装的信息的足够访问,以便可以在它们上执行算法。所有修改都通过容器接口进行,以保证不变量。

例如,如果您考虑sort算法的要求。对于STL使用的(通常)实现,它需要(来自容器):

  • 有效访问给定索引处的项目:随机访问
  • 交换两个项目的能力:不是关联的

因此,任何提供随机访问且不是关联的容器(理论上)都适合通过(例如)快速排序算法进行有效排序。

C ++中满足此要求的容器是什么?

  • 基本的C阵列
  • deque
  • vector

如果你注意这些细节,你可以写的任何容器。

为每个人重写(复制/粘贴/调整)sort会是浪费,不是吗?

请注意,例如,有一个std::list::sort方法。为什么?因为std::list不提供随机访问(非正式地myList[4]不起作用),因此来自算法的sort不适合。


19
投票

我使用的标准是,如果一个函数可以通过成员函数显着更有效地实现,那么它应该是一个成员函数。 ::std::sort不符合这个定义。事实上,在外部和内部实施它没有任何效率差异。

通过实现成员(或朋友)功能来提高效率意味着通过了解类的内部状态可以大大提高效率。

界面设计的一部分技术是找到最小的成员函数集,这样您可能希望在对象上执行的所有操作都可以合理有效地实现。而且这个集合不应该支持不应该在类上执行的操作。因此,您不能只实现一堆getter和setter函数并将其称为好。


11
投票

我认为这个规则的原因是通过使用成员函数,你可能会偶然地依赖于类的内部。改变类的状态不是问题。真正的问题是,如果修改类中的某些私有属性,则需要更改的代码量。保持类(公共方法)的接口尽可能小,既减少了在这种情况下需要做的工作量,又减少了对私有数据做一些奇怪的事情的风险,使得实例处于不一致状态。

AtoMerZ也是对的,非成员非朋友函数也可以模板化并重用于其他类型。

顺便说一句,你应该购买你的有效C ++的副本,这是一本很棒的书,但不要总是遵守本书的每一项。面向对象设计既有良好的实践(来自书籍等)和经验(我认为它也是用有效的C ++编写的)。


3
投票

所以,第一个问题,他们不应该成员吗?

不,这不遵循。在惯用的C ++类设计中(至少在Effective C ++中使用的习语中),非成员非友元函数扩展了类接口。它们可以被视为该类的公共API的一部分,尽管它们不需要也不具有对该类的私有访问权。如果这个设计是OOP的某些定义的“非OOP”那么,好的,惯用的C ++不是那个定义的OOP。

将相同的推理延伸到vector类中的其他一些函数

确实如此,标准容器的一些成员函数可以是自由函数。例如,vector::push_back是根据insert定义的,当然可以在没有私有访问类的情况下实现。在这种情况下,push_back是一个抽象概念的一部分,BackInsertionSequence,矢量实现。这些通用概念贯穿于特定类的设计,因此,如果您正在设计或实现可能会影响函数放置位置的通用概念。

当然,标准的某些部分可能应该是不同的,例如std::string has way too many member functions。但是已经完成了什么,这些类是在人们真正适应我们现在称之为现代C ++风格之前设计的。这个课程无论哪种方式都有效,所以你可以从担心差异中获得如此多的实际好处。


2
投票

动机很简单:保持一致的语法。随着课程的发展或使用,将出现各种非会员便利功能;例如,你不想修改类接口以将类似toUpper的东西添加到字符串类中。 (在std::string的情况下,当然,你不能。)斯科特担心的是,当发生这种情况时,你最终会得到不一致的语法:

s.insert( "abc" );
toUpper( s );

通过仅使用自由函数,根据需要声明它们的朋友,所有函数都具有相同的语法。另一种方法是在每次添加便利功能时修改类定义。

我并不完全相信。如果一个类设计得很好,它具有基本功能,用户可以清楚地知道哪些功能是该基本功能的一部分,哪些是附加的便利功能(如果有的话)。在全球范围内,字符串是一种特殊情况,因为它旨在用于解决许多不同的问题;我无法想象很多课程都是如此。


2
投票

各种想法:

  • 当非成员通过类的公共API工作时,这很好,因为它减少了以下代码的数量: 需要仔细监控以确保不变量, 如果重新设计对象的实现,则需要更改。
  • 如果这还不够好,非成员仍然可以成为friend
  • 编写非成员函数通常不太方便,因为成员不是隐含在范围内,但如果考虑程序演变: 一旦存在非成员函数并且意识到相同的功能对其他类型有用,通常很容易将函数转换为模板并使其不仅可用于两种类型,而且可用于任意未来类型。换句话说,非成员模板允许比运行时多态/虚拟调度更灵活的算法重用:模板允许称为duck typing。 具有有用成员函数的现有类型鼓励剪切和粘贴到其他类似行为的类型,因为转换函数以进行重用的大多数方法要求每个隐式成员访问都对特定对象进行显式访问,对于程序员来说,这将是一个更长的30秒以上....
  • 成员函数允许使用object.function(x, y, z)表示法,其中恕我直言,非常方便,富有表现力和直觉。它们在许多IDE中的发现/完成功能方面也能更好地工作。
  • 作为成员和非成员函数的分离可以帮助传达类的基本性质,它是不变量和基本操作,并且逻辑上对附加组件和可能的特殊“便利”特征进行分组。考虑Tony Hoare的智慧: “构建软件设计有两种方法:一种方法是使其变得如此简单以至于显然没有缺陷,另一种方法是使其变得如此复杂以至于没有明显的缺陷。第一种方法要困难得多“。 在这里,非成员使用并不一定要困难得多,但您必须更多地考虑如何访问成员数据和私有/受保护方法以及原因,以及哪些操作是基础。这样的灵魂搜索也可以通过成员函数改进设计,它更容易变得懒惰: - /。
  • 由于非成员功能在复杂性方面进行了扩展或获得了额外的依赖关系,因此可以将这些功能移动到单独的头文件和实现文件中,甚至库中,因此核心功能的用户只需“付费”使用他们想要的部件。

(Omnifarious的答案是必读的,如果它对你来说是新的三倍。)


1
投票

我认为sort不是作为成员函数实现的,因为它被广泛使用,不仅适用于向量。如果他们将它作为一个成员函数,那么每次使用它时,他们都必须重新实现它。所以我认为这是为了更容易实现。

© www.soinside.com 2019 - 2024. All rights reserved.