当前位置: 首页 > news >正文

01【C++ 入门基础】命名空间/域

文章目录

  • 引言
  • 命名空间是什么?
    • 定义命名空间
      • 嵌套命名空间
  • namespace如何做到隔离命名的?
    • 使用命名空间
      • 1.using namespace 展开命名空间
      • 2.使用using引入命名空间中的某个成员
      • 3.命名空间名称+作用域限定符
  • 拓展
  • 总结


引言

通过00【C++ 入门基础】前言得知,C++是为了解决C语言在面对大型项目的局限而诞生:

C语言面对的现实工程问题(复杂性、可维护性、可扩展性、安全性)

C语言中,无论是相同文件还是不同的文件之间,都不可以出现重名的函数、变量,变量、函数和结构体的名称将都存在于全局作用域中,会出现命名冲突或名字污染等问题,
C++中的namespace关键字的出现就是针对这种问题的。

命名冲突问题:

#include<stdlib.h>
int rand = 0;
int main()
{return 0;
}

报错:“rand”: 重定义;以前的定义是“函数”
原因:我们自定义的rand和stdlib.h库中的rand函数命名冲突了,导致编译器分不清。

命名空间是什么?

命名空间顾名思义,是隔离命名区域的一个空间,在C语言中,没有命名空间的概念,那么只能在整个全局域随便命名,伴随的便是各种变量、函数、结构体的命名冲突,在C++中,通过命名空间,隔离出一个空间,不同空间的命名互不影响,这样就解决了命名重复的问题。
在这里插入图片描述

定义命名空间

定义命名空间,需要使用到namespace关键字,后面跟命名空间的名字,然后接一对{}即可,{}中即为命名空间的成员。

namespace mySpace
{int rand = 0;
}
namespace yourSpace
{int rand(){return 6;}
}
int main()
{return 0;
}

嵌套命名空间

命名空间之间还可以嵌套,做到将我们一个命令空间的区域再划分小。

namespace mySpace
{int rand = 0;namespace yourSpace{int rand(){return 6;}}
}
int main()
{return 0;
}

如果同一个工程中允许存在多个相同名称的命名空间怎么办?
编译器最后会将它们合成同一个命名空间。那么如果我们的同名命名空间中,又有命名冲突怎么办?
很简单,用命名空间嵌套。

namespace如何做到隔离命名的?

其实我们namespace XXX{}的方式,指定的是一个命名空间域,域(Scope)​​ 是一个核心概念,它定义了程序中标识符(如变量、函数、类名)的可见性与生命周期范围。我们目前常见的域有:

1.局部域
​范围​:整个程序文件(跨文件通过 extern 共享)。
​问题​:所有符号直接暴露,​无隔离​(C 语言的局限)。
2.全局域
​范围​:函数、循环、条件语句内部的 {} 块。
​生命周期​:变量在退出作用域时自动销毁。

C++ 的命名空间(Namespace)本质是一种特殊的域,用于解决全局作用域下的命名冲突问题。

使用命名空间

命名空间中成员该如何使用呢?

命名空间的使用有三种方式:
1.using namespace 展开命名空间
2.使用using引入命名空间中的某个成员
3.命名空间名称+作用域限定符

namespace mySpace
{int a = 0;
}int main()
{int b = a;  //报错“a”: 未声明的标识符return 0;
}

命名空间就是用来解决全局作用域下的命名冲突,所以直接访问肯定不可行。

1.using namespace 展开命名空间

#include<stdio.h>
namespace mySpace
{int a = 0;
}using namespace mySpace;  //展开mySpace命名空间域,相当于将我们mySpace的域和全局域"接壤"//那么我们全局中就可以使用了.
int main()
{int b = a;return 0;
}

那么现在我们就大概可以知道,我们一般使用STL库(standard template library标准模板库)的时候,常写的“using namespace std;”是什么,“std”其实就是我们STL库的一个命名空间。

但是注意,直接将我们的空间展开,那么那么我们一个程序就可以存在同名的不同成员,但是这时候访问就会有二义性问题,因为编译器并不知道访问哪个,所以我们在开发大型项目的时候,还是建议我们使用的第2、3个方法。

#include<stdio.h>
namespace mySpace
{int a = 0;
}using namespace mySpace;int a = 2;
int main()
{printf("%d",a);  // 报错:“a”: 不明确的符号return 0;
}

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

#include<stdio.h>
namespace mySpace
{int a = 0;int b = 1;
}using mySpace::a;//指定引入变量a.int main()
{int c;c = a;       //a引入了可以正常使用.c = b;       //b未引入,所以报错:“b”: 未声明的标识符return 0;
}

我们using指定引入变量,就相当于在两个域中(全局域和命名空间域)中,开了一个特殊通道,允许我们的编译器可以单独找到这个变量。
当我们的程序中,使用到一个的函数、变量多次,那么我们就可以使用这种指定引入的方式:

#include<stdio.h>
#include<iostream>
using std::cout;
int main()
{cout << "a";cout << "a";cout << "a";cout << "a";cout << "a";cout << "a";cout << "a";return 0;
}

3.命名空间名称+作用域限定符

一般的编译器自动域搜索过程,是一个由内到外的过程,局部作用域=>再…外层的作用域=>全局作用域=>using namespace引入的命名空间域。
而我们的作用域限定符“: :”,是C++解决命名冲突的关键机制,可以让我们指定某个域访问变量:

#include<stdio.h>
namespace mySpace
{int a = 0;int b = 1;
}
int a = 1;
int main()
{int a = 2;printf("%d", a);           //由编译器自动根据作用域顺序找到printf("%d", ::a);         //不加命名空间名的就是指定全局域访问printf("%d", mySpace::a);  //指定访问mySpace中的变量return 0;
}

拓展

头文件include展开和命名空间域using展开完全不是同一个东西!!
域是域,库是库。

C++也可以用“.h”的头文件,但是为什么C++中的头文件没有.h?
在比较老的C++版本中,当时也没有命名空间,之后C++的更新中才有了命名空间,便把很多的东西全加入到C++库中(std命名空间中),C++标准库为了与C语言区分开来,并引入命名空间的概念,所以给后续的头文件都设计不带“.h”。
然后也把一些C语言的库,带“.h”的头文件,都重新命名,去掉“.h”,在最前面加上“c”(stdio.h头文件 => cstdio),注意,即便是现在,我们所有带“.h”的C语言头文件,也都是可以正常使用的,但是它也还是会污染命名空间,而改变之后的带“c”的头文件,内部已经全部被std命名空间封装了。

总结

1.命名空间解决的是全局变量的重名问题。
2.命名空间使用有三种方法(域作用限定符是关键)。
3.“c”式的C语言库,是经过命名空间std封装的库。


本文章为作者的笔记和心得记录,顺便进行知识分享,有任何错误请评论指点:)。

相关文章:

  • 8、做中学 | 四年级下期 Golang运算符
  • [论文阅读] 人工智能 + 软件工程 | AI 与敏捷开发的破局之路:从挫败到成功的工作坊纪实
  • Git 使用规范与命令使用场景详解
  • 【嵌入式ARM汇编基础】-ELF文件格式内部结构详解(二)
  • C语言再出发:2025年AI时代的关键语言
  • JavaWeb学习——day9(图书管理系统初级)
  • Day 2 学习主题「面向对象 + Pythonic 风格」
  • Linux服务器部署Leantime与cpolar构建低成本团队协作环境
  • 数据分享:汽车行业-汽车属性数据集
  • 英特尔汽车业务败走中国,喊出“All in”才过两个月
  • 测试方法的分类
  • 香港维尔利健康科技集团推出AI辅助医学影像训练平台,助力医护人才数字化转型
  • aws(学习笔记第四十七课) codepipeline-docker-build
  • 深入解析设备管理系统新趋势:物联网与云原生驱动的智能化实践
  • 软件测试之基于博客系统项目的性能测试
  • 大数据赋能智慧城市:从数据洪流到科学规划的“智慧之匙”
  • 互联网医院系统源码解析:如何实现视频问诊、电子处方等核心功能?
  • 详解零拷贝
  • 面试150 验证回文串
  • 七天学会SpringCloud分布式微服务——02——第一个微服务项目