《C++之那些年踩过的坑(一)》
C++之那些年踩过的坑(一)
作者:刘俊延(Alinshans)
本系列文章针对我在写C++代码的过程中,尤其是做自己的项目时,踩过的各种坑。以此作为给自己的警惕。
转载请注明原文来自: http://www.cnblogs.com/GodA/p/6501505.html
这一篇就讲点简单的东西,名称空间。
一、谨慎在全局范围使用 using namespace XXX
注意,是在全局范围使用 using namespace XXX,除非写个玩具,否则最好不要这样。之前很喜欢在文件开头就使用 using namespace std,因为可以省很多事。后来我在写 MyTinySTL 这个项目的时候,写测试时为了方便,用了 using namespace std,结果惨不忍睹,大量的名称冲突。当使用 using namepsace xxx 的时候,会把 xxx 这个名称空间里的内容引入到当前空间,如果很不幸,有相同的函数,类等等,就会造成冲突。然而如果碰巧没有冲突,那也只是暂时的,谁知道以后会加入一些什么东西呢?
比如,你写了这样的东西:
#include <iostream>
#include <vector> namespace your
{ template <typename T>
class vector
{
public:
vector() = default;
~vector() = default;
private:
T* data;
}; } using namespace std;
using namespace your; int main()
{
vector<int> v;
}
很明显,这是编译不过的,同时引入了 std 和 your 两个名称空间,且都有一个 vector 模板类,编译器是不知道要使用哪个的。当然,这个太明显了没有人会这么做。但是,如果代码多了呢?文件多了呢?一不小心在哪个地方用了,也不知道会不会 #include 的时候一层一层的也间接的弄了进来。即使你在某个局部的名称空间xxx内声明了 using namespace std,但是也不能保证哪天谁用你的代码,然后为了方便使用你的东西,using namespace xxx,然后又间接引入了一个 std 导致爆炸。
有很多更好的解决方法。比直接在全局使用 using 声明好一点的是,在局部使用,并且只声明需要的部分。比如有时候我们经常在一个函数/作用域内使用 std::cout, std::endl,那么或许可以这样:
void test()
{
using std::cout;
using std::endl; cout << "1+1=" << + << endl;
cout << "1+2=" << + << endl;
cout << "1+3=" << + << endl;
// ...
}
上面这样写当然是没问题了啦,不过我现在习惯,并推崇的还是显式声明,一来这样可以更清晰的知道,用的是哪里来的东西,而且几乎放多久都不会错,二来多敲几个键而已,不是什么麻烦事。以上说了很多,都是以 std 为例,对其它的,也应该保持同样的态度,不禁止,但是要谨慎。不过还是养成好习惯比较好,那么就把 using namespace std; 写成 std::xxx 吧。
放一段 C++ Coding Standards 里面的话:
Summary
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. (The second rule follows from the first, because headers can never know what other header #includes might appear after them.)
Discussion
In short: You can and should use namespace using declarations and directives liberally in your implementation files after #include directives and feel good about it. Despite repeated assertions to the contrary, namespace using declarations and directives are not evil and they do not defeat the purpose of namespaces. Rather, they are what make namespaces usable.
在放一句 Google C++ Style Guide 里的一条:
You may not use a using-directive to make all names from a namespace available.
二、关于匿名 namespace
关于 unnamed namespace,cppreference 如是说到:
Its members have potential scope from their point of declaration to the end of the translation unit, and have internal linkage.
也就是说,现在,在匿名空间里的成员,具有内部链接。现在来说,匿名空间跟 static 就没区别了。不过依然要注意,在同一层次中,可以有多个匿名空间,不过这些匿名空间会被整合成一个,所以不能像这样写,会报重定义:
#include <iostream> namespace
{
void foo() { std::cout << "" << "\n"; }
} namespace
{
void foo() { std::cout << "" << "\n"; }
} int main()
{
foo();
}
这很容易理解。在不同空间内的匿名函数,就是不同的啦,比如这样就可以通过了:
#include <iostream> namespace
{
void foo() { std::cout << "" << "\n"; }
} namespace n1
{
namespace
{
void foo() { std::cout << "" << "\n"; }
}
} int main()
{
foo();
n1::foo();
}
在 Google C++ Style Guide 中,这样讲到匿名namespace:
When definitions in a
.cc
file do not need to be referenced outside that file, place them in an unnamed namespace or declare themstatic
. Do not use either of these constructs in.h
files.Use of internal linkage in
.cc
files is encouraged for all code that does not need to be referenced elsewhere. Do not use internal linkage in.h
files.
他们鼓励在实现文件中,把那些不需要外部引用的东西放进匿名空间中。陈硕大大在他的 CppPractice 中,第一个提到的就是慎用匿名空间。我觉得,对于他说的不利之处,现在来看,主要还是是第一点。因为匿名namespace里的东西是匿名的,所以万一以后有一天想引用它了,也说不准。其实还是用个具体名称,也不麻烦。对于那些实现细节,或者不希望暴露的,我喜欢扔进一个 namespace details{} 或者什么 namespace impl {} 里。
今天就先谈这么多。总结一下我自己的观点:
- 使用 using namespace xxx xxx::yyy
- 一般情况下,都使用具名 namespace
《C++之那些年踩过的坑(一)》的更多相关文章
- 简单物联网:外网访问内网路由器下树莓派Flask服务器
最近做一个小东西,大概过程就是想在教室,宿舍控制实验室的一些设备. 已经在树莓上搭了一个轻量的flask服务器,在实验室的路由器下,任何设备都是可以访问的:但是有一些限制条件,比如我想在宿舍控制我种花 ...
- 利用ssh反向代理以及autossh实现从外网连接内网服务器
前言 最近遇到这样一个问题,我在实验室架设了一台服务器,给师弟或者小伙伴练习Linux用,然后平时在实验室这边直接连接是没有问题的,都是内网嘛.但是回到宿舍问题出来了,使用校园网的童鞋还是能连接上,使 ...
- 外网访问内网Docker容器
外网访问内网Docker容器 本地安装了Docker容器,只能在局域网内访问,怎样从外网也能访问本地Docker容器? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动Docker容器 ...
- 外网访问内网SpringBoot
外网访问内网SpringBoot 本地安装了SpringBoot,只能在局域网内访问,怎样从外网也能访问本地SpringBoot? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装Java 1 ...
- 外网访问内网Elasticsearch WEB
外网访问内网Elasticsearch WEB 本地安装了Elasticsearch,只能在局域网内访问其WEB,怎样从外网也能访问本地Elasticsearch? 本文将介绍具体的实现步骤. 1. ...
- 怎样从外网访问内网Rails
外网访问内网Rails 本地安装了Rails,只能在局域网内访问,怎样从外网也能访问本地Rails? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动Rails 默认安装的Rails端口 ...
- 怎样从外网访问内网Memcached数据库
外网访问内网Memcached数据库 本地安装了Memcached数据库,只能在局域网内访问,怎样从外网也能访问本地Memcached数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装 ...
- 怎样从外网访问内网CouchDB数据库
外网访问内网CouchDB数据库 本地安装了CouchDB数据库,只能在局域网内访问,怎样从外网也能访问本地CouchDB数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动Cou ...
- 怎样从外网访问内网DB2数据库
外网访问内网DB2数据库 本地安装了DB2数据库,只能在局域网内访问,怎样从外网也能访问本地DB2数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动DB2数据库 默认安装的DB2 ...
- 怎样从外网访问内网OpenLDAP数据库
外网访问内网OpenLDAP数据库 本地安装了OpenLDAP数据库,只能在局域网内访问,怎样从外网也能访问本地OpenLDAP数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动 ...
随机推荐
- [javaSE] 集合框架(迭代器)
当我们创建一个集合以后,可以直接使用system.out.println()来打印这个集合,但是,我们需要可以对每个元素进行操作,所以,这里需要使用迭代器来遍历集合 迭代器其实就是集合取出元素的方式 ...
- memcached 细究(二)
{ CentOS ping命令 分布式部署服务器时用到ping命令 #ping -c 4 192.168.16.1 //ping4次后结束. } 使用telnet 查看memcached 运行状态 ...
- K:线性表的实现—顺序表
所谓顺序表,就是顺序存储的线性表.顺序存储就是用一组地址连续的存储单元依次存放线性表中各个数据元素的存储结构. 线性表中所有数据元素的类型是相同的,所以每一个数据元素在存储器中占用相同的大小的空间.假 ...
- 理解webpack4.splitChunks之maxInitialRequests
maxInitialRequests是splitChunks里面比较难以理解的点之一,它表示允许入口并行加载的最大请求数,之所以有这个配置也是为了对拆分数量进行限制,不至于拆分出太多模块导致请求数量过 ...
- Javascript获取For循环所用时间
第一种: let tOne = new Date().getTime(); let n = new Date(); let hour = n.getHours() < 10 ? "0& ...
- docker中使用nginx容器代理其他容器
Nginx is an HTTP and reverse proxy server, a mail proxy server, and a generic TCP/UDP proxy server, ...
- 一步一步pwn路由器之路由器环境修复&&rop技术分析
前言 本文由 本人 首发于 先知安全技术社区: https://xianzhi.aliyun.com/forum/user/5274 拿到路由器的固件后,第一时间肯定是去运行目标程序,一般是web服务 ...
- [算法练习]Add Two Numbers
题目说明: You are given two linked lists representing two non-negative numbers. The digits are stored in ...
- Mybatis学习第四天——一对一&&一对多
两表关系: 1.Mybatis中一对一关系 <!-- 两表联查,通过相同属性user_id left join 表示以左边的表为主表 --> <select id="fin ...
- Android画廊效果
Android画廊效果 前言:Gallery是一个内部元素控件,可以水平滚动,并且可以把当前选择的子元素定位在它中心的布局组件:画廊Gallery一般用来显示可左右移动图片的列表(具体请看实例). 效 ...