C++ using 编译指令与名称冲突问题

 更新时间:2022年11月21日 10:05:51   作者:木三百川  
using 编译指令由名称空间名和它前面的关键字 using namespace 组成,它使名称空间中的所有名称都可用,而不需要使用作用域解析运算符,这篇文章主要介绍了C++ using 编译指令与名称冲突,需要的朋友可以参考下

using 编译指令由名称空间名和它前面的关键字 using namespace 组成,它使名称空间中的所有名称都可用,而不需要使用作用域解析运算符。与 using 声明不同的是,using 编译指令会进行名称解析,在一些时候名称空间的变量会被同区域声明的同名变量隐藏,不会出现名称冲突的报错。但在另一些情况下,使用 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 编译指令时可能遇到的各种情况列表如下,表中的最后一列是指在作用域的重合区域使用变量时是否会引发名称冲突。

到此这篇关于C++ using 编译指令与名称冲突的文章就介绍到这了,更多相关C++ using 编译指令内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

场景同名全局变量 pail同名局部变量pail另一名称空间 Rose 的同名变量 pailusing 编译指令是否名称冲突
1存在冲突
2存在存在冲突
3存在不冲突
4存在存在不冲突
5存在冲突
6存在存在不冲突
7存在存在存在不冲突
8不冲突

到此这篇关于C++ using 编译指令与名称冲突问题的文章就介绍到这了,更多相关C++ using 编译指令内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 使用C++绘制GDI位图的基本编写实例

    使用C++绘制GDI位图的基本编写实例

    这篇文章主要介绍了使用C++绘制GDI位图的基本编写实例,一般来说适用于Windwos下的C++的GUI编程,需要的朋友可以参考下
    2015-12-12
  • C++算法计时器的实现示例

    C++算法计时器的实现示例

    本文主要介绍了C++算法计时器的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-05-05
  • 详解桶排序算法的思路及C++编程中的代码实现

    详解桶排序算法的思路及C++编程中的代码实现

    桶排序即是先把每个桶中的元素进行排序然后遍历桶依次列出元素的算法,桶排序在元素较少的情况下很高效,以下我们就来详解桶排序算法的思路及C++编程中的代码实现:
    2016-07-07
  • C语言中字符和字符串处理(ANSI字符和Unicode字符)

    C语言中字符和字符串处理(ANSI字符和Unicode字符)

    这篇文章主要介绍了C语言与C++中字符和字符串处理(ANSI字符和Unicode字符)的详细内容,非常的全面,这里推荐给大家,希望大家能够喜欢。
    2015-03-03
  • C++ 类和对象基础篇

    C++ 类和对象基础篇

    类是创建对象的模板,一个类可以创建多个对象,每个对象都是类类型的一个变量;创建对象的过程也叫类的实例化。每个对象都是类的一个具体实例(Instance),拥有类的成员变量和成员函数
    2020-01-01
  • Qt编写地图之实现经纬度坐标纠偏

    Qt编写地图之实现经纬度坐标纠偏

    地图应用中都涉及到一个问题就是坐标纠偏的问题,这个问题的是因为根据地方规则保密性要求不允许地图厂商使用标准的GPS坐标,而是要用国家定义的偏移标准。本文将详细讲解如何在Qt中实现经纬度坐标纠偏,需要的可以参考一下
    2022-03-03
  • C语言学习之链表的实现详解

    C语言学习之链表的实现详解

    链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的。这篇文章主要介绍了C语言中链表的实现,需要的可以参考一下
    2022-11-11
  • 深入解读C++ 内联函数inline|nullptr

    深入解读C++ 内联函数inline|nullptr

    内联函数:用** inline 修饰的函数叫做内联函数,编译时C++编译器会在调用的地方展开内联函数**,这样调用内联函数就需要创建栈桢,就提高效率了,这篇文章给大家介绍C++ 内联函数inline|nullptr的相关知识,感兴趣的朋友跟随小编一起看看吧
    2024-07-07
  • C语言完数的实现示例

    C语言完数的实现示例

    C语言中的完数指的是一个正整数,本文主要介绍了C语言完数,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-05-05
  • c语言中unsigned修饰符的使用

    c语言中unsigned修饰符的使用

    在C语言中,unsigned是一种无符号整数修饰符,本文主要介绍了c语言中unsigned修饰符的使用,具有一定的参考价值,感兴趣的可以了解一下
    2023-11-11

最新评论