C ++命名空间函数

时间:2012-05-08 05:07:13

标签: c++ namespaces

我是一名Java开发人员,我对C ++很陌生。我需要实现某种实用类,我正在考虑将这些方法实现为static。但是,我遇到了关于命名空间函数与静态方法的this stackoverflow question,并且显然名称空间函数是首选方法。所以我想知道如果有关于如何实现命名空间功能的任何文章或示例。例如,我应该如何在头文件中声明命名空间函数?头文件是否只包含类头文件等函数定义,实现应该在cpp文件中,还是我应该直接在头文件中实现函数?

基本上,我正在尝试实现一个应用程序来解析包含一些命令的文本文件。所以我正在考虑实现静态辅助方法来处理文本处理。例如readCommand(字符串行)。如果我的方向错误,请告诉我。感谢

3 个答案:

答案 0 :(得分:25)

  

我应该如何在头文件中声明命名空间函数?

namespace MON {
// extern:
t_ret func(const t_param& pValue);
// 'inline':
inline t_ret inline_func(const t_param& pValue) { ... }
} // << MON

  

头文件是否只包含类头文件等函数定义,实现应该在cpp文件中,还是我应该直接在头文件中实现函数?

取决于您是否希望它们(可能)内联或导出。这通常归结为最小化依赖性。

扩展出口或内联:

你经常喜欢extern函数来最小化c ++中的依赖。这相当于在类方法中将定义与声明分开:

<强> file.hpp

namespace MON {
// extern:
t_ret func(const t_param& pValue);
} // << MON

<强> file.cpp

#include "hefty_stuff.hpp"

MON::t_ret MON::func(const t_param& pValue) { ... }
但是,有时候定义在某些情况下是可见的,通常是为了提高性能,或者当你知道大小很重要并且标题不包含在很多地方时,它有时是至关重要的。因此,inline变体也是一种选择。

内联函数仍然可以导出,并且可以按要求内联 - 但是,任何内联函数副本都可以合并(具体来说,实现可以自由地假设所有定义都相同,并且函数的任何副本都是不必要的)。

使用导出的定义,您可以有选择地限制(或隔离)您的包含依赖项。也就是说,#include "hefty_stuff.hpp"不需要在标题中使用file.hpp中的函数。


  

基本上,我正在尝试实现一个应用程序来解析包含一些命令的文本文件。所以我正在考虑实现静态辅助方法来处理文本处理。

嗯,这里应该避免使用static。 c ++使用one-definition-rule。 static只会导致许多不必要的副本。此外,匿名命名空间是c static函数的c ++方法:

namespace {
t_ret func(const t_param& pValue) { ... }
} // << anon

注意:匿名命名空间也可能导致不必要的副本。你将它们用作静态函数的替代的原因是你想要或者需要偏离one-definition-rule,并且不想在可能被“解析”的范围中声明符号。


关于template<>声明的最后一点。使用模板,定义必须在使用的位置可见,除非您的编译器支持extern模板。对于模板,您可以通过多种方式完成定义可见性。通常,人们只需简单地声明定义,或为定义添加标题,该标题包含在标题的末尾或根据需要。使用模板,不需要声明函数inline以避免多个定义错误。

答案 1 :(得分:17)

您可以在标题中声明函数:

namespace A {
    void foo();
}

并在.cpp中实现:

namespace A {
  void foo() { std::cout << "foo!"; }
}

您还可以将实现放在标题中,确保将其声明为inline以避免违反one definition rule

namespace A {
    inline void foo() { std::cout << "foo()!"; }
}

请注意,将实现放在标头中意味着客户端代码对实现具有编译依赖性,以及用于实现的标头。在上面的例子中,客户端代码现在依赖于头部,如果我们做一些微不足道的事情,比如在打印输出中添加感叹号,我们需要重新编译,而不是重新链接所有客户端代码。 / p>

将模板函数的实现放在标题或标题包含的文件中是非常重要的,这些不能放在.cpp中:

namespace B {
  template <class T>
  inline void foo(const T& t) { std::cout << t.name() << "\n"; }
}

答案 2 :(得分:2)

我应该如何在头文件中声明命名空间函数?

namespace YourNamespace
{
    void fun1();
    void fun2();
}

头文件是否只包含类头文件等函数定义,实现应该在cpp文件中,还是我应该直接在头文件中实现函数?

如果命名空间中的函数是静态的,则可以在头文件中实现函数,或者必须在cpp文件中实现。