当前位置:   article > 正文

C++命名空间namespace

C++命名空间namespace

前言:

         在C/C++中,变量、函数和后面要学到的类都是大量存在的,这些变量、函数和类的名称将都存在于全局作用域中,可能会导致很多冲突。在C++中,为了解决这一问题,C++添加了namespace这一关键字,对标识符的名称进行本地化, 以避免命名冲突或名字污染,namespace关键字的出现就是针对这种问题的。

一、问题引入

如果我们写了以下代码

  1. #include<stdio.h>
  2. int n(int a, int b)
  3. {
  4. return a + b;
  5. }
  6. int main()
  7. {
  8. int n = 1;
  9. int b = n(1, 2);
  10. return 0;
  11. }

我们会发现编译器会报错,原因是我们定义了一个名字为n的函数后,我们又定义了一个名字为n的变量,所以编译器无法区分二者,所以会报错。为了避免这种情况的发生,我们可以引入命名空间的概念,即在某个空间中,命名有其特定的含义。

二、创建命名空间

创建命名空间的格式如下:

  1. namespace space_name
  2. {
  3. //name
  4. }

在命名空间中,我们可以定义变量、函数、类/结构体,注意,在命名空间中不能定义两个名字相同的变量、函数、类。

例如:我们对上面的代码进行修改

  1. namespace fun
  2. {
  3. int n(int a, int b)
  4. {
  5. return a + b;
  6. }
  7. }
  8. namespace num
  9. {
  10. int n;
  11. }
  12. int main()
  13. {
  14. //这里的n是num命名空间中的n
  15. num::n = 1;
  16. cout << num::n << endl;
  17. //这里的n是fun命名空间中的n函数
  18. cout << fun::n(1, 2) << endl;
  19. return 0;
  20. }

我们在fun命名空间中定义了一个名为n的函数,又在num命名空间中定义了一个名为n的变量。这样我们就可以同时使用变量n和函数n。

三、命名空间成员的使用

命名空间成员的使用有三种方法:

1.加命名空间名称及作用域限定符

  1. int main()
  2. {
  3. //这里的n是num命名空间中的n
  4. num::n = 1;
  5. cout << num::n << endl;
  6. //这里的n是fun命名空间中的n函数
  7. cout << fun::n(1, 2) << endl;
  8. return 0;
  9. }

在成员前面加空间名,对成员进行使用。

注意:这种方式访问时,必须每次在每个空间成员的前面加上空间名。

2.使用using将命名空间中某个成员引入

  1. namespace num
  2. {
  3. int x;
  4. int n;
  5. }
  6. int main()
  7. {
  8. using num::n;
  9. //这里的n是num命名空间中的n
  10. n = 1;
  11. cout << n << endl;
  12. //这里的n是fun命名空间中的n函数
  13. cout << fun::n(1, 2) << endl;
  14. return 0;
  15. }

在使用成员之前,加上using+空间名,后面再次使用该成员时就不需再加空间名。

注意:在使用这种方法时,只有被引入的成员后续使用不需要再次引入,其余变量仍然需要引入。

3.使用using namespace 命名空间名称 引入

  1. using namespace num;
  2. int main()
  3. {
  4. //这里的n是num命名空间中的n
  5. n = 1;
  6. cout << n << endl;
  7. //这里的n是fun命名空间中的n函数
  8. cout << fun::n(1, 2) << endl;
  9. return 0;
  10. }

在使用成员之前,加上using namespace +空间名,将命名空间完全展开,后面使用该空间中的所有成员都不需要再加空间名。

四、命名空间的嵌套问题

命名空间的嵌套其实就是在命名空间中再定义一个命名空间,在不同的子命名空间中又可以定义同名的成员。

我们就可以设计以下命名空间num:其中包括两个子命名空间,

  1. namespace num
  2. {
  3. namespace num1
  4. {
  5. int x;
  6. int n;
  7. }
  8. namespace num2
  9. {
  10. int x;
  11. int n;
  12. }
  13. }

在调用的时候,同上面一样,只需要展开两层,就能够引入子命名空间中的成员。

  1. void test1()
  2. {
  3. num::num1::n = 1;
  4. cout << num::num1::n << endl;
  5. }
  6. void test2()
  7. {
  8. using num::num1::n;
  9. n = 1;
  10. cout << n << endl;
  11. }
  12. void test3()
  13. {
  14. using namespace num::num1;
  15. n = 1;
  16. cout << n << endl;
  17. }
  18. int main()
  19. {
  20. test1();
  21. test2();
  22. test3();
  23. return 0;
  24. }

我们同样可以使用以上三种方法来引用空间中的成员。

五、命名空间展开作用域问题

命名空间的展开只在其所在的作用周期生效。

比如说,我们在函数内展开某个命名空间,当我们出了该命名空间,我们便不能直接引用空间的成员。

例如以下代码,就会报错:

  1. namespace num
  2. {
  3. int n;
  4. }
  5. void test1()
  6. {
  7. using namespace num;
  8. }
  9. int Add(int x)
  10. {
  11. return n + x;
  12. }
  13. int main()
  14. {
  15. cout << Add(3) << endl;
  16. return 0;
  17. }

原因是num命名空间只在test1中展开,并没有在Add函数中展开,Add中无法直接使用n。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Gausst松鼠会/article/detail/151174
推荐阅读
相关标签
  

闽ICP备14008679号