C++命名空间实例解析

命名空间是C++非常重要的概念,本文就以实例形式对其进行深入分析,具体内容如下:

通常来说,在C++中,命名空间(namespace)的目的是为了防止名字冲突。每个命名空间是一个作用域,在所有命名空间之外,还存在一个全局命名空间(global namespace),全局命名空间以隐式的方式声明,它并没有名字。在命名空间机制中,原来的全局变量,就是位于全局命名空间中(可以用::member的形式表示)。

一、定义命名空间

1、每个命名空间都是一个作用域

和其他作用域类似,在命名空间中的每个名字必须表示唯一实体,而在不同命名空间中,可以有相同名字的成员。

2、命名空间可以是不连续的

命名空间可以定义在几个不同的部分:

namespace nsp {
  /* …… */
}  // 命名空间作用域后面无须分号

如果之前没有名为nsp的命名空间定义,则上述代码创建一个新的命名空间;否则,上述代码打开已经存在的命名空间添加一些新的成员。

3、命名空间是可以嵌套的

嵌套的命名空间是指定义在其他命名空间中的命名空间。嵌套的命名空间是一个嵌套的作用域,内层命名空间声明的名字将隐藏外层命名空间声明的同名成员:

int x = 20;
namespace outer {
  int x = 10;
  namespace inner {
    int z = x;
  }
} 

int main()
{
  std::cout << outer::inner::z; // 输出10
  return 0;
}

注意,通常我们不把#include头文件放在命名空间内部。

4、未命名的命名空间

未命名的命名空间(unnamed namespace)是指关键字namespace后紧跟花括号括起来的一系列声明语句。未命名的命名空间中定义的变量拥有静态生命周期:它们在第一次使用前创建,并且直到程序结束才销毁。在标准C++引入命名空间的概念之前,程序需要将名字声明成static以使其对于整个文件有效,但是,现在在文件中进行静态声明的做法已经被C++标准取消了,取而代之的是使用未命名的命名空间。

一个未命名的命名空间可以在某个给定的文件内不连续,但是不能跨越多个文件。也就是说,未命名的命名空间仅在特定的文件内部有效,其作用范围不会横跨多个不同的文件。另外,由于未命名的命名空间它没有名字,所以其中定义的名字的作用域与该命名空间所在的作用域相同:

int i;  // i的全局声明
namespace {
  int i;
}
i = 10; // 错误,二义性
namespace local {
  namespace {
    int i;
  }
}
local::i = 42; // 正确

二、使用命名空间

对命名空间中成员的引用,需要使用命名空间的作用域运算符(::)。但是,像namespace_name::member_name这样使用命名空间的成员非常烦琐,我们需要使用一些其他的更简便的方法。

1、命名空间的别名

有些命名空间的名字很长或者命名空间嵌套了很多层,我们可以为其设定一个较短的同义词,也就是别名:

namespace cln = cpluslus_learning_namespace;
namespace Qlib = outer::inner::QueryLib;

2、using声明

一条using声明(using declaration)语句一次只引入命名空间的一个成员:

using 命名空间名::[命名空间名::……]成员名;  // 例如 using OLib::List;

3、using指示

using指示(using directive)和using声明不同的地方是,我们无法控制哪些名字是可见的,因为using指示会使得某个特定的命名空间中所有的名字都可见:

using namespace std;  // 引入命名空间std

使用命名空间主要是为了防止名字冲突,如果随意使用using指示注入命名空间的所有名字,将重新引入名字冲突的问题。另外,using声明和using指示在作用域上有区别:using声明是将一个成员引入当前命名空间作用域内;using指示是将所有成员引入当前和上一层命名空间作用域内:

namespace nsp {  // 命名空间
  int i=16, j=15;
} 

int j = 0;  // 全局变量 

int main()
{
  using namespace nsp;  // 如果使用j,将在::j和nsp::j之间产生冲突
  std::cout << j;
  return 0;
}

如果将using namespace nsp;改成using声明using nsp::j;则会正确输出15。

(0)

相关推荐

  • C++编程中的命名空间基本知识讲解

    命名空间是一个声明性区域,为其内部的标识符(类型.函数和变量等的名称)提供一个范围.命名空间用于将代码组织到逻辑组中,还可用于避免名称冲突,尤其是在基本代码包括多个库时.命名空间范围内的所有标识符彼此可见,而没有任何限制.命名空间之外的标识符可通过使用每个标识符的完全限定名(例如 std::vector<std::string> vec;)来访问成员,也可通过单个标识符的 using 声明 (using std::string) 或命名空间中所有标识符的 using 指令 (C++) (usi

  • C++命名空间实例解析

    命名空间是C++非常重要的概念,本文就以实例形式对其进行深入分析,具体内容如下: 通常来说,在C++中,命名空间(namespace)的目的是为了防止名字冲突.每个命名空间是一个作用域,在所有命名空间之外,还存在一个全局命名空间(global namespace),全局命名空间以隐式的方式声明,它并没有名字.在命名空间机制中,原来的全局变量,就是位于全局命名空间中(可以用::member的形式表示). 一.定义命名空间 1.每个命名空间都是一个作用域 和其他作用域类似,在命名空间中的每个名字必须

  • PHP关键特性之命名空间实例详解

    命名空间主要是为了解决代码中类和函数可能存在冲突的问题,而这个特性其他语言一早就有,PHP则是姗姗来迟,它的出现催生了 PSR-4 的诞生,从而也催生了 Composer 的兴起,所以是非常重要的特性. 命名空间的定义 命名空间是一个容器,这个容器主要是为了识别其下的类和函数.一旦定义了命名空间,它下面的代码就属于这个命名空间了,所以命名空间的定义要在代码的最开始行. 对于同一个包来说,同一个命名空间或者子命名空间的代码没有必要在一个 PHP 文件中定义,子命名空间下的代码是为了完成特定模块的工

  • python如何重载模块实例解析

    本文首先介绍了Python中的模块的概念,谈到了一个模块往往由多个模块组成,然后通过具体实例,分析了模块重载的相关内容,具体介绍如下. 模块是Python程序架构的一个核心概念,较大的程序往往以多个模块文件的形式呈现,一个模块被设计成主文件或顶层文件,用来启动整个Python程序.每个以.py为后缀的Python源代码文件都是一个模块,其他文件可通过"导入"读取这个模块的内容.从一般意义上讲,模块就是变量名的封装.如写一个模块test.py,包含一个两个变量名name.age. nam

  • Spring Bean装载方式代码实例解析

    这篇文章主要介绍了Spring Bean装载方式代码实例解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 Bean的装配方式 Bean的装配可以理解为依赖关系注入 基于XML的装配 a) 设值注入 i.要求: Bean 类必须提供一个默认的无参构造方法. Bean 类必须为需要注入的属性提供对应的setter方法. b) 构造注入 package com.itheima.assemble; import java.util.List; pub

  • Java反射技术详解及实例解析

    前言 相信很多人都知道反射可以说是Java中最强大的技术了,它可以做的事情太多太多,很多优秀的开源框架都是通过反射完成的,比如最初的很多注解框架,后来因为java反射影响性能,所以被运行时注解APT替代了,java反射有个开源框架jOOR相信很多人都用过,不过我们还是要学习反射的基础语法,这样才能自己写出优秀的框架,当然这里所讲的反射技术,是学习Android插件化技术.Hook技术等必不可少的! 一.基本反射技术   1.1 根据一个字符串得到一个类 getClass方法 String nam

  • C++命名空间实例详解

    一个中大型软件往往由多名程序员共同开发,会使用大量的变量和函数,不可避免地会出现变量或函数的命名冲突.当所有人的代码都测试通过,没有问题时,将它们结合到一起就有可能会出现命名冲突. 例如小李和小韩都参与了一个文件管理系统的开发,它们都定义了一个全局变量 fp,用来指明当前打开的文件,将他们的代码整合在一起编译时,很明显编译器会提示 fp 重复定义(Redefinition)错误. 为了解决合作开发时的命名冲突问题,C++ 引入了命名空间(Namespace)的概念.请看下面的例子: namesp

  • JavaScript正则表达式校验与递归函数实际应用实例解析

    JS递归函数(菲波那切数列) 实例解析: 一组数字:0 1 1 2 3 5 8 13 0 1 2 3 4 5 6 7 sl(0)=0; sl(1)=1; sl(2)=sl(0)+sl(1); sl(3)=sl(1)+sl(2); function sl(i){ if(i==0){ return 0; }else if(i==1){ return 1; }else{ return sl(i-1)+sl(i-2); } } 正则表达式检验 //校验是否全由数字组成 function isDigit(

  • python数据类型判断type与isinstance的区别实例解析

    在项目中,我们会在每个接口验证客户端传过来的参数类型,如果验证不通过,返回给客户端"参数错误"错误码. 这样做不但便于调试,而且增加健壮性.因为客户端是可以作弊的,不要轻易相信客户端传过来的参数. 验证类型用type函数,非常好用,比如 >>type('foo') == str True >>type(2.3) in (int,float) True 既然有了type()来判断类型,为什么还有isinstance()呢? 一个明显的区别是在判断子类. type(

  • XML轻松学习手册(5)XML实例解析

    第五章:XML实例解析 提纲: 一:实例效果 二:实例解析 1.定义新标识. 2.建立XML文档. 3.建立相应的HTML文件. XML在不同领域有着广泛的应用,比如在科技领域的MathML,无线通信应用的WML,在网络图象方面的SVG等等,我们这里侧重讨论XML在web上的应用.XML在web上应用主要是利用其强大的数据操作能力.一般用XML配合javascript和asp等服务器端程序,可以实现网络上几乎所有的应用需求. 考虑讲解方便,我们在下面介绍一个简单的实例,不包含服务器端程序.目的在

  • MYSQL子查询和嵌套查询优化实例解析

    查询游戏历史成绩最高分前100 Sql代码 SELECT ps.* FROM cdb_playsgame ps WHERE ps.credits=(select MAX(credits) FROM cdb_playsgame ps1 where ps.uid=ps1.uid AND ps.gametag=ps1.gametag) AND ps.gametag='yeti3' GROUP BY ps.uid order by ps.credits desc LIMIT 100; Sql代码 SEL

随机推荐