C++ using 编译指令与名称冲突
using 编译指令:它由名称空间名和它前面的关键字 using namespace 组成,它使名称空间中的所有名称都可用,而不需要使用作用域解析运算符。在全局声明区域中使用 using 编译指令,将使该名称空间的名称全局可用;在函数或代码块中使用 using 编译指令,将使其中的名称在该函数或代码块中可用。当包含 using 声明的最小声明区域中已经声明了和名称空间中相同的名称时,若仍使用 using 声明导入该名称空间的同名名称,则这两个名称将会发生冲突,编译器会报错。与 using 声明不同的是,using 编译指令会进行名称解析,在一些时候名称空间的变量会被同区域声明的同名变量隐藏,不会出现名称冲突的报错。但在另一些情况下,使用 using 编译指令仍会出现名称冲突的报错,下面对此进行总结,测试所用的环境为 Microsoft Visual Studio 2019 以及 QT 5.9.2 MinGW 32bit。
1 using 编译指令与同名全局变量
结论:若仅存在同名全局变量,不存在同名局部变量,使用 using 编译指令后,在作用域的重合区域使用变量时一定会引发名称冲突。除非在同名全局变量声明前的代码块中使用,但这时是因为同名变量的作用域不重合,而非 using 编译指令名称解析的功劳。
1.1 在同名全局变量声明前使用
若在同名全局变量声明前的代码块中使用,由于作用域不重合,一定不会引发名称冲突,因此只需测试在同名全局变量声明前的全局区中使用 using 编译指令的效果。测试程序如下:(出现名称冲突报错)
#include <iostream>
//自定义名称空间
namespace Jack {
double pail = 1;
}
//在同名全局变量声明前使用
using namespace Jack;
//在全局名称空间中定义变量
double pail = 2;
//测试
int main()
{
using namespace std;
//使用
cout << pail << endl;
cout << ::pail << endl;
cout << Jack::pail << endl;
return 0;
}
运行结果如下:

1.2 在同名全局变量声明后的全局区中使用
测试程序如下:(出现名称冲突报错)
#include <iostream>
//自定义名称空间
namespace Jack {
double pail = 1;
}
//在全局名称空间中定义变量
double pail = 2;
//在同名全局变量声明后使用
using namespace Jack;
//测试
int main()
{
using namespace std;
//使用
cout << pail << endl;
cout << ::pail << endl;
cout << Jack::pail << endl;
return 0;
}
运行结果如下:

1.3 在同名全局变量声明后的代码块中使用
测试程序如下:(出现名称冲突报错)
#include <iostream>
//自定义名称空间
namespace Jack {
double pail = 1;
}
//同名全局变量声明
double pail = 2;
//测试
int main()
{
using namespace std;
//使用
using namespace Jack;
cout << pail << endl;
cout << ::pail << endl;
cout << Jack::pail << endl;
return 0;
}
运行结果如下:

2 using 编译指令与同名局部变量
结论:若仅存在同名局部变量,不存在同名全局变量,使用 using 编译指令将会进行名称解析,不会引发名称冲突,但在代码块中,同名局部变量将隐藏名称空间中的变量。
2.1 在同名局部变量声明前的全局区中使用
测试程序如下:(运行成功)
#include <iostream>
//自定义名称空间
namespace Jack {
double pail = 1;
}
//在同名局部变量声明前的全局区中使用
using namespace Jack;
//测试
int main()
{
using namespace std;
//同名局部变量
double pail = 2;
//使用
cout << pail << endl; //结果为2
cout << ::pail << endl; //结果为1
cout << Jack::pail << endl; //结果为1
return 0;
}
运行结果如下:

2.2 在同名局部变量声明前的代码块中使用
测试程序如下:(运行成功)
#include <iostream>
//自定义名称空间
namespace Jack {
double pail = 1;
}
//测试
int main()
{
using namespace std;
//在同名局部变量声明前的代码块中使用
using namespace Jack;
//同名局部变量
double pail = 2;
//使用
cout << pail << endl; //结果为2
cout << Jack::pail << endl; //结果为1
return 0;
}
运行结果如下:

2.3 在同名局部变量声明后使用
若在同名局部变量声明后的全局区中使用,由于作用域不重合,一定不会引发名称冲突,因此只需测试在同名局部变量声明后的代码块中使用 using 编译指令的效果。测试程序如下:(运行成功)
#include <iostream>
//自定义名称空间
namespace Jack {
double pail = 1;
}
//测试
int main()
{
using namespace std;
//同名局部变量
double pail = 2;
//在同名局部变量声明后的代码块中使用
using namespace Jack;
//使用
cout << pail << endl; //结果为2
cout << Jack::pail << endl; //结果为1
return 0;
}
运行结果如下:

3 不同名称空间中的同名变量
结论:若不同名称空间中存在同名变量,不存在同名全局变量以及同名局部变量,使用 using 编译指令后,在作用域的重合区域使用变量时一定会引发名称冲突。
3.1 using 编译指令位置都在全局区中
测试程序如下:(出现名称冲突报错)
#include <iostream>
//自定义名称空间
namespace Jack {
double pail = 1;
}
namespace Rose {
double pail = 2;
}
//都在全局区中
using namespace Jack;
using namespace Rose;
//测试
int main()
{
using namespace std;
//使用
cout << pail << endl;
return 0;
}
运行结果如下:

3.2 using 编译指令位置都在代码块中
测试程序如下:(出现名称冲突报错)
#include <iostream>
//自定义名称空间
namespace Jack {
double pail = 1;
}
namespace Rose {
double pail = 2;
}
//测试
int main()
{
using namespace std;
//都在代码块中
using namespace Jack;
using namespace Rose;
//使用
cout << pail << endl;
return 0;
}
运行结果如下:

3.3 using 编译指令位置不同区
测试程序如下:(出现名称冲突报错)
#include <iostream>
//自定义名称空间
namespace Jack {
double pail = 1;
}
namespace Rose {
double pail = 2;
}
//Jack位于全局区中
using namespace Jack;
//测试
int main()
{
using namespace std;
//Rose位于代码块中
using namespace Rose;
//使用
cout << pail << endl;
return 0;
}
运行结果如下:

4 多个同名变量共存
结论:若名称空间中的变量、同名全局变量、同名局部局部变量三者同时存在,using 编译指令的使用位置不会影响名称解析的结果,且不会引发名称冲突,这正是 using 编译指令进行名称解析的效果。
4.1 在同名全局变量声明前的全局区中使用
测试程序如下:(运行成功)
#include <iostream>
//自定义名称空间
namespace Jack {
double pail = 1;
}
//在同名全局变量声明前的全局区中使用
using namespace Jack;
//同名全局变量
double pail = 2;
//测试
int main()
{
using namespace std;
//同名局部变量
double pail = 3;
//使用
cout << pail << endl; //结果为3
cout << ::pail << endl; //结果为2
cout << Jack::pail << endl; //结果为1
return 0;
}
运行结果如下:

4.2 在同名全局变量声明后的全局区中使用
测试程序如下:(运行成功)
#include <iostream>
//自定义名称空间
namespace Jack {
double pail = 1;
}
//同名全局变量
double pail = 2;
//在同名全局变量声明后的全局区中使用
using namespace Jack;
//测试
int main()
{
using namespace std;
//同名局部变量
double pail = 3;
//使用
cout << pail << endl; //结果为3
cout << ::pail << endl; //结果为2
cout << Jack::pail << endl; //结果为1
return 0;
}
运行结果如下:

4.3 在同名局部变量声明前的代码块中使用
测试程序如下:(运行成功)
#include <iostream>
//自定义名称空间
namespace Jack {
double pail = 1;
}
//同名全局变量
double pail = 2;
//测试
int main()
{
using namespace std;
//在同名局部变量声明前的代码块中使用
using namespace Jack;
//同名局部变量
double pail = 3;
//使用
cout << pail << endl; //结果为3
cout << ::pail << endl; //结果为2
cout << Jack::pail << endl; //结果为1
return 0;
}
运行结果如下:

4.4 在同名局部变量声明后的代码块中使用
测试程序如下:(运行成功)
#include <iostream>
//自定义名称空间
namespace Jack {
double pail = 1;
}
//同名全局变量
double pail = 2;
//测试
int main()
{
using namespace std;
//同名局部变量
double pail = 3;
//在同名局部变量声明后的代码块中使用
using namespace Jack;
//使用
cout << pail << endl; //结果为3
cout << ::pail << endl; //结果为2
cout << Jack::pail << endl; //结果为1
return 0;
}
运行结果如下:

5 总结
通过上述多个测试,可以得到以下结论:
- 若仅存在同名全局变量,不存在同名局部变量,使用
using编译指令后,在作用域的重合区域使用变量时一定会引发名称冲突。 - 若仅存在同名局部变量,不存在同名全局变量,使用
using编译指令将会进行名称解析,不会引发名称冲突,但在代码块中,同名局部变量将隐藏名称空间中的变量。 - 若不同名称空间中存在同名变量,不存在同名全局变量以及同名局部变量,使用
using编译指令后,在作用域的重合区域使用变量时一定会引发名称冲突。 - 若名称空间中的变量、同名全局变量、同名局部局部变量三者同时存在,
using编译指令的使用位置不会影响名称解析的结果,且不会引发名称冲突,这正是using编译指令进行名称解析的效果。
以 Jack 名称空间中的 pail 变量为例,将使用 using 编译指令时可能遇到的各种情况列表如下,表中的最后一列是指在作用域的重合区域使用变量时是否会引发名称冲突。
| 场景 | 同名全局变量 pail |
同名局部变量 pail |
另一名称空间 Rose 的同名变量 pail |
using 编译指令是否名称冲突 |
|---|---|---|---|---|
| 1 | 存在 | 无 | 无 | 冲突 |
| 2 | 存在 | 无 | 存在 | 冲突 |
| 3 | 无 | 存在 | 无 | 不冲突 |
| 4 | 无 | 存在 | 存在 | 不冲突 |
| 5 | 无 | 无 | 存在 | 冲突 |
| 6 | 存在 | 存在 | 无 | 不冲突 |
| 7 | 存在 | 存在 | 存在 | 不冲突 |
| 8 | 无 | 无 | 无 | 不冲突 |
C++ using 编译指令与名称冲突的更多相关文章
- using 声明与编译指令
using std::cout; // using 声明 using namespace std; // using 编译指令,导入std里面的所有名称 一般使用using 声明,它只会导入指定的名称 ...
- iOS中的预编译指令的初步探究
目录 文件包含 #include #include_next #import 宏定义 #define #undef 条件编译 #if #else #endif #if define #ifdef #i ...
- 【JSP】JSP基础学习记录(一)—— 基础介绍以及3个编译指令
序: 从实现到现在一直是以.net为主,但偶尔也会参与一些其他语言的项目.最近需要对一个Java Web项目进行二次开发,一直没学习过JSP所以买了几本书自学试试.参考资料为<轻量级Java E ...
- C++ 中的名称冲突之 "y1"
已经是第二次遇到这个问题了: #include <bits/stdc++.h> using namespace std; ); ][N][][N]; int x1, x2, y1, y2; ...
- 二、JSP的3个编译指令,7个动作指令,9个内置对象
JSP 3个编译指令 1) page指令(基本不需要用到,使用默认即可) 主要属性: 1.Language:指定脚本所采用的语言类型,现在只支持java 2.Extends:定义当前jsp产生的se ...
- Delphi 预编译指令 的用法
A.3 使用条件编译指令条件编译指令是非常重要的编译指令,他控制着在不同条件下(例如,不同的操作系统)产生不同的代码.条件编译指令是包含在注释括号之内的,如下表所示. ...
- Delphi 预编译指令
<Delphi下深入Windows核心编程>(附录A Delphi编译指令说明)Delphi快速高小的编译器主要来自Object PASCAL的严谨,使用Delphi随时都在与编译器交流, ...
- JSP学习(2)---四种基本语法与三种编译指令
JSP的异常可以不处理,即使是checked异常. 四种基本语法: jsp声明,jsp注释,jsp表达式,jsp脚本 三种编译指令: page,include,taglib 下面是具体的练习. sho ...
- c/c++中的预编译指令总结
预处理指令提供按条件跳过源文件中的节.报告错误和警告条件,以及描绘源代码的不同区域的能力.使用术语“预处理指令”只是为了与 C 和 C++ 编程语言保持一致.在 C# 中没有单独的预处理步骤:预处理指 ...
- jsp 的 3 个编译指令
JSP 的编译指令是通知 JSP 引擎的消息,它不直接生成输出. 常见的编译指令有如下三个: 1.page:该指令是针对当前页面的指令 2.include:用于指定包含另一个页面 3.taglib:用 ...
随机推荐
- 【读书笔记】C#高级编程 第十五章 反射
(一)在运行期间处理和检查代码 自定义特性允许把自定义元数据与程序元素关联起来.反射是一个普通术语,它描述了在运行过程中检查和处理程序元素的功能.例如,反射允许完成的任务: 枚举类型的成员 实例化新对 ...
- dp-背包模型
一:01背包问题模型 1 题目: 有一个箱子容量为 V,同时有 n 个物品,每个物品有一个体积(正整数). 要求 n 个物品中,任取若干个装入箱内,使箱子的剩余空间为最小. 输入格式 第一行是一个整数 ...
- 引擎之旅 Chapter.1 高分辨率时钟
目录 游戏中的时间线 真实时间线 游戏时间线 全局时钟的实现方式 我们如何理解时间.在现实生活中,时间就是一个有方向的直线.从一个无穷远到另一个无穷远.用数学去抽象地思考,它就是一个从无穷小到无穷大的 ...
- day36-IO流03
JavaIO流03 4.常用的类02 4.4节点流和处理流 4.4.1基本介绍 节点流可以从一个特定的数据源读写数据,如FileReader.FileWriter 数据源就是存放数据的地方,比如文件. ...
- Lua 支持虚函数的解决方案
概述 lua本身没有提供类似C++虚函数机制,调用的父类方法调用虚函数可能会出现问题. 问题分析 分析这段代码和输出 local Gun = {} -- 示例,实际应用还要考虑构造,虚表等情况 fun ...
- 使用Elasticsearch Operator快速部署Elasticsearch集群
转载自:https://www.qikqiak.com/post/elastic-cloud-on-k8s/ 随着 kubernetes 的快速发展,很多应用都在往 kubernetes 上面迁移,现 ...
- 五、frp内网穿透客户端frpc.ini各配置参数详解
[必须]标识头[common]是不可或缺的部分 [必须]frps服务端IPserver_addr = 0.0.0.00.0.0.0为FRP服务端IP,客户端要填写为服务端已配置的对应的IP,或者是服务 ...
- Elasticsearch:Pinyin 分词器
Elastic的Medcl提供了一种搜索Pinyin搜索的方法.拼音搜索在很多的应用场景中都有被用到.比如在百度搜索中,我们使用拼音就可以出现汉字: 对于我们中国人来说,拼音搜索也是非常直接的.那么在 ...
- MySQL 的七种日志总结
文章转载自:https://mp.weixin.qq.com/s/ewv7HskHvH3O7kFyOmoqgw 一.MySQL 日志分类 日志类别 说明 备注 错误日志 错误日志记录了当MySQL启动 ...
- 第四章:Django表单 - 3:Django表单字段汇总
Field.clean(value)[source] 虽然表单字段的Field类主要使用在Form类中,但也可以直接实例化它们来使用,以便更好地了解它们是如何工作的.每个Field的实例都有一个cle ...