我个人使用c++的风格总是把类声明放在包含文件中,定义放在。cpp文件中,这很像Loki对c++头文件的回答,代码分离。不可否认,我喜欢这种风格的部分原因可能与我花了这么多年来编写Modula-2和Ada有关,这两者都有类似的规范文件和主体文件方案。

我有一个同事,他比我更了解c++,他坚持认为所有c++声明都应该尽可能地在头文件中包含定义。他并不是说这是一种有效的替代风格,甚至不是一种稍微更好的风格,而是说这是一种新的普遍接受的风格,现在每个人都在使用c++。

我不像以前那么灵活了,所以在我看到更多的人和他在一起之前,我并不急于加入他的行列。那么这个习语到底有多普遍呢?

只是给答案一些结构:现在的方式™,非常常见,有点常见,不常见,还是疯狂?


当前回答

这难道不是取决于系统的复杂性和内部约定吗?

目前,我正在研究一个非常复杂的神经网络模拟器,我期望使用的公认风格是:

classname.h中的类定义 classnameCode.h中的类代码 classname.cpp中的可执行代码

这将用户构建的模拟从开发人员构建的基类中分离出来,在这种情况下效果最好。

但是,如果有人在图形应用程序或其他目的不是为用户提供代码库的应用程序中这样做,我会感到惊讶。

其他回答

我认为你的同事是对的,只要他没有进入这个过程,在头部写可执行代码。 我认为,正确的平衡是遵循GNAT Ada所指出的路径,其中.ads文件为其用户和其子程序包提供了完全适当的接口定义。

顺便说一句,Ted,你有没有在这个论坛上看到最近关于Ada绑定到CLIPS库的问题,这个问题是你几年前写的,现在已经没有了(相关的网页现在已经关闭了)。即使使用的是旧的Clips版本,对于愿意在Ada 2012程序中使用Clips推理引擎的人来说,这个绑定也是一个很好的开始示例。

为了增加更多的乐趣,你可以添加.ipp文件,其中包含模板实现(包含在.hpp中),而.hpp包含接口。

除了模板化的代码(取决于项目,这可能是大部分或少数文件),还有正常的代码,这里最好将声明和定义分开。在需要的地方也提供前向声明——这可能会影响编译时间。

你的同事错了,通常的方法是把代码放在。cpp文件(或任何你喜欢的扩展名)中,并在头文件中声明。

将代码放在头文件中偶尔会有一些好处,这可以让编译器更聪明地进行内联。但与此同时,它会破坏编译时间,因为所有代码在每次被编译器包含时都必须被处理。

最后,当所有的代码都是头部时,循环对象关系(有时是需要的)通常是令人讨厌的。

总之,你是对的,他是错的。

编辑:我一直在思考你的问题。在一种情况下,他说的是真的。模板。许多较新的“现代”库,如boost,大量使用模板,通常是“头文件”。但是,只有在处理模板时才应该这样做,因为这是处理模板时的唯一方法。

编辑:有些人想要更多的澄清,这里有一些关于写“只有头”代码的缺点的想法:

If you search around, you will see quite a lot of people trying to find a way to reduce compile times when dealing with boost. For example: How to reduce compilation times with Boost Asio, which is seeing a 14s compile of a single 1K file with boost included. 14s may not seem to be "exploding", but it is certainly a lot longer than typical and can add up quite quickly when dealing with a large project. Header only libraries do affect compile times in a quite measurable way. We just tolerate it because boost is so useful.

此外,还有许多事情不能仅在头文件中完成(甚至boost也有一些库,您需要链接到某些部分,如线程、文件系统等)。一个主要的例子是你不能在只有头的库中有简单的全局对象(除非你求助于讨厌的单例),因为你会遇到多个定义错误。注意:c++ 17的内联变量将使这个特殊的例子在未来可行。

最后一点,当使用boost作为仅头代码的示例时,经常会忽略一个巨大的细节。

Boost是库,而不是用户级代码。所以它不会经常变化。在用户代码中,如果你把所有东西都放在头文件中,每一个小的改变都会导致你不得不重新编译整个项目。这是对时间的巨大浪费(对于那些在不同的编译器之间不进行更改的库来说,情况并非如此)。当你在头文件/源文件之间拆分时,更好的是,使用前向声明来减少包含,你可以在一天内节省数小时的重新编译时间。

如果这个新方法真的是方法,我们可能在我们的项目中遇到了不同的方向。

因为我们尽量避免在头文件中出现不必要的东西。这包括避免头级联。头文件中的代码可能需要包含一些其他头文件,而这些头文件又需要另一个头文件,以此类推。如果我们被迫使用模板,我们尽量避免在标题中使用太多模板。

如果适用,我们还使用“不透明指针”模式。

通过这些实践,我们可以比大多数同行更快地构建。是的……更改代码或类成员不会导致巨大的重构。

我个人在头文件中这样做:

// class-declaration

// inline-method-declarations

我不喜欢将方法的代码与类混合在一起,因为我发现快速查找东西很痛苦。

我不会把所有的方法都放在头文件中。编译器(通常)不能内联虚拟方法,(可能)只内联没有循环的小方法(完全取决于编译器)。

在类中执行方法是有效的…但从可读性的角度来看,我不喜欢它。将方法放在头文件中确实意味着,如果可能的话,它们将被内联。