“using namespace” in c++ headers

后端 未结 9 2193
失恋的感觉
失恋的感觉 2020-11-22 01:32

In all our c++ courses, all the teachers always put using namespace std; right after the #includes in their .h files. This seems to me

相关标签:
9条回答
  • 2020-11-22 01:49

    I believe you can use 'using' in C++ headers safely if you write your declarations in a nested namespace like this:

    namespace DECLARATIONS_WITH_NAMESPACES_USED_INCLUDED
    {
        /*using statements*/
    
        namespace DECLARATIONS_WITH_NO_NAMESPACES_USED_INCLUDED
        {
            /*declarations*/
        }
    }
    
    using namespace DECLARATIONS_WITH_NAMESPACES_USED_INCLUDED::DECLARATIONS_WITH_NO_NAMESPACES_USED_INCLUDED;
    

    This should include only the things declared in 'DECLARATIONS_WITH_NO_NAMESPACES_USED_INCLUDED' without the namespaces used. I have tested it on mingw64 compiler.

    0 讨论(0)
  • 2020-11-22 01:50

    With regards to "Is there some way to undo [a using declaration]?"

    I think it is useful to point out that using declarations are affected by scope.

    #include <vector>
    
    {   // begin a new scope with {
        using namespace std;
        vector myVector;  // std::vector is used
    }   // end the scope with }
    
    vector myOtherVector;   // error vector undefined
    std::vector mySTDVector // no error std::vector is fully qualified
    

    So effectively yes. By limiting the scope of the using declaration its effect only lasts within that scope; it is 'undone' when that scope ends.

    When the using declaration is declared in a file outside of any other scope it has file-scope and affects everything in that file.

    In the case of a header file, if the using declaration is at file-scope this will extend to the scope of any file the header is included in.

    0 讨论(0)
  • 2020-11-22 01:51

    You should definitely NOT use using namespace in headers for precisely the reason you say, that it can unexpectedly change the meaning of code in any other files that include that header. There's no way to undo a using namespace which is another reason it's so dangerous. I typically just use grep or the like to make sure that using namespace isn't being called out in headers rather than trying anything more complicated. Probably static code checkers flag this too.

    The header should include just the headers that it needs to compile. An easy way to enforce this is to always include each source file's own header as the first thing, before any other headers. Then the source file will fail to compile if the header isn't self-contained. In some cases, for example referring to implementation-detail classes within a library, you can use forward declarations instead of #include because you have full control over the definition of such forward declared class.

    I'm not sure I would call it common, but it definitely shows up once in a while, usually written by new programmers that aren't aware of the negative consequences. Typically just a little education about the risks takes care of any issues since it's relatively simple to fix.

    0 讨论(0)
  • 2020-11-22 01:57

    Item 59 in Sutter and Alexandrescu's "C++ Coding Standards: 101 Rules, Guidelines, and Best Practices":

    59. Don’t write namespace usings in a header file or before an #include.

    Namespace usings are for your convenience, not for you to inflict on others: Never write a using declaration or a using directive before an #include directive.

    Corollary: In header files, don't write namespace-level using directives or using declarations; instead, explicitly namespace-qualify all names.

    A header file is a guest in one or more source files. A header file that includes using directives and declarations brings its rowdy buddies over too.

    A using declaration brings in one buddy. A using directive brings in all the buddies in the namespace. Your teachers' use of using namespace std; is a using directive.

    More seriously, we have namespaces to avoid name clash. A header file is intended to provide an interface. Most headers are agnostic of what code may include them, now or in the future. Adding using statements for internal convenience within the header foists those convenient names on all the potential clients of that header. That can lead to name clash. And it's just plain rude.

    0 讨论(0)
  • 2020-11-22 02:03

    You are right. And any file should only include the headers needed by that file. As for "is doing things wrong common in real world projects?" - oh, yes!

    0 讨论(0)
  • 2020-11-22 02:05

    You are right that using namespace in header is dangerous. I do not know a way how to undo it. It is easy to detect it however just search for using namespace in header files. For that last reason it is uncommon in real projects. More experienced coworkers will soon complain if someone does something like it.

    In real projects people try to minimize the amount of included files, because the less you include the quicker it compiles. That saves time of everybody. However if the header file assumes that something should be included before it then it should include it itself. Otherwise it makes headers not self-contained.

    0 讨论(0)
提交回复
热议问题