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

cpp中的namespace详解

namespace的作用主要是为了避免名字冲突和组织代码。

命名空间在C++中是一个非常重要的特性,它帮助开发者更好地管理代码和避免潜在的冲突。

具体来说,它有以下几个主要用途

  1. 避免名字冲突
    在大型项目中可能会有很多个类、函数或变量使用相同的名称。使用命名空间可以将这些名称分组,从而避免冲突。
#include <iostream>namespace ProjectA {void display() {std::cout << "Project A Function" << std::endl;}
}namespace ProjectB {void display() {std::cout << "Project B Function" << std::endl;}
}int main() {ProjectA::display(); // 调用Project A的函数ProjectB::display(); // 调用Project B的函数return 0;
}
  1. 代码组织
    命名空间有助于逻辑上组织代码,使代码结构更加清晰。可以帮助开发者将相关的功能组织在一起,便于管理和维护。
#include <iostream>namespace Math {void add(int a, int b) {std::cout << "Sum: " << (a + b) << std::endl;}void subtract(int a, int b) {std::cout << "Difference: " << (a - b) << std::endl;}
}namespace Utils {void printGreeting() {std::cout << "Welcome to the Math Program!" << std::endl;}void printFarewell() {std::cout << "Thank you for using the Math Program!" << std::endl;}
}
int main() {Utils::printGreeting();Math::add(5, 3);Math::subtract(5, 3);Utils::printFarewell();return 0;
}
  1. 提供作用域
    命名空间为其中定义的标识符提供了一个独立的作用域。即使在同一个文件中,可以定义多个同名的函数或变量,只要它们位于不同的命名空间中。
#include <iostream>namespace Math {int value = 10; // Math命名空间内的变量void display() {std::cout << "Math Value: " << value << std::endl;}
}namespace Science {int value = 20; // Science命名空间内的变量void display() {std::cout << "Science Value: " << value << std::endl;}
}int main() {Math::display();    // 调用Math命名空间的display函数Science::display(); // 调用Science命名空间的display函数// 可以使用不同命名空间的变量std::cout << "Accessing Math value: " << Math::value << std::endl;std::cout << "Accessing Science value: " << Science::value << std::endl;return 0;
}
  1. 嵌套命名空间
    C++支持嵌套命名空间,可以进一步组织代码,避免名称冲突。
#include <iostream>namespace Outer {namespace Inner {void display() {std::cout << "Hello from Inner Namespace!" << std::endl;}int value = 42;}void show() {std::cout << "Hello from Outer Namespace!" << std::endl;}
}int main() {Outer::show();               // 调用外部命名空间的函数Outer::Inner::display();     // 调用嵌套命名空间的函数std::cout << "Inner value: " << Outer::Inner::value << std::endl; // 访问嵌套命名空间的变量return 0;
}
  1. 使用简化
    可以使用using声明来简化命名空间内标识符的访问。
#include <iostream>namespace Math {int add(int a, int b) {return a + b; // 加法函数}int subtract(int a, int b) {return a - b; // 减法函数}
}using namespace Math; // 使用整个 Math 命名空间int main() {int sum = add(5, 3);           // 直接调用函数,无需前缀int difference = subtract(5, 3); // 直接调用函数std::cout << "Sum: " << sum << std::endl;           // 输出结果std::cout << "Difference: " << difference << std::endl; // 输出结果return 0;
}

在不同的cpp文件中使用相同的 namespace xxx;

分析:

1、共享命名空间:所有文件都使用命名空间 xxx,避免命名冲突,同时保持代码的整洁。
2、功能分离:日志记录和实用工具功能分开,易于管理和扩展。
3、模块化设计:可以独立编译和链接各个文件,增强了代码的可维护性。
4、一致性:通过相同的命名空间,开发者可以清楚地识别出相关功能。C++会将它们合并。

logger/Logger (h/cpp)
#ifndef LOGGER_H
#define LOGGER_H#include <string>namespace xxx {class Logger {public:void log(const std::string& message);};
}#endif====================================================#include "Logger.h"
#include <iostream>namespace xxx {void Logger::log(const std::string& message) {std::cout << "Log: " << message << std::endl;}
}
utilities/Utils (h/cpp)
#ifndef UTILS_H
#define UTILS_Hnamespace xxx {class Utils {public:static void printHello();};
}#endif====================================================#include "Utils.h"
#include <iostream>namespace xxx {void Utils::printHello() {std::cout << "Hello from Utils!" << std::endl;}
}
main.cpp
#include "logger/Logger.h"
#include "utilities/Utils.h"int main() {xxx::Logger logger;logger.log("This is a log message.");xxx::Utils::printHello();return 0;
}
拓展
使用相同的 namespace xxx时函数名和参数都相同会出现什么情况呢?

如果在两个不同的c++文件中使用相同的命名空间xxx,并且里面的函数名相同,编译时会出现重定义错误,这是因为统一命名空间内不允许有重复定义的标识符。

使用相同的 namespace xxx时函数名相同和参数类型不相同会出现什么情况呢?

如果参数不同,这种情况称为函数重载。c++允许在同一命名空间中重载函数。及时它们的名称相同,只要是参数列表不同(包括参数数量或类型)

如果不使用namespace会出现什么情况呢?

file1.cpp

#if 0 //未使用namespace
#include <iostream>int setting = 42; // 全局变量void printSetting() {std::cout << "file1 setting: " << setting << std::endl;
}void display() {std::cout << "Display from file1" << std::endl;
}
#else //使用namespace
namespace file1 {int setting = 42;void printSetting() {std::cout << "file1 setting: " << setting << std::endl;}
}
namespace FileA {void display() {std::cout << "Display from fileA" << std::endl;}
}
#endif

file2.cpp

#include <iostream>
#if 0 //未使用namespace
int setting = 100; // 同名全局变量,覆盖了 file1.cpp 中的 settingvoid printUserSetting() {std::cout << "file2 setting: " << setting << std::endl;
}void display() {std::cout << "Display from fileB" << std::endl;
}
#else //使用namespace
namespace file2 {int setting = 100;void printUserSetting() {std::cout << "file2 setting: " << setting << std::endl;}
}
namespace fileB {void display() {std::cout << "Display from fileB" << std::endl;}
}
#endif

main.cpp

#include <iostream>
#if 0 //未使用namespace
void display(); // 声明void printSetting();    // 声明void printUserSetting(); // 声明
#else //使用namespace
namespace file1 {void printSetting();
}namespace file2 {void printUserSetting();
}namespace fileA {void display();
}namespace fileB {void display();
}
#endifint main() {
#if 0 //未使用namespaceprintSetting();       // 期望输出 file1 setting: 42printUserSetting();   // 期望输出 file2 setting: 100display(); // 编译时错误:不确定调用哪个cpp下的display函数
#else //使用namespacefile1::printSetting();       // 输出 file1 setting: 42file2::printUserSetting();     // 输出 file2 setting: 100fileA::display(); // 调用 fileA 命名空间中的函数fileB::display(); // 调用 fileB 命名空间中的函数
#endifreturn 0;
}

1、命名冲突:不同文件或库中相同名称的类、函数或变量可能导致编译错误,提示重定义。因为编译器无法区分它们。
2、可读性降低:代码的结构可能变得混乱,特别是在大型项目中,所有的标识符(如函数名、变量名等)都在一个全局范围内,难以追踪各个功能的来源。增加了理解和维护的难度。
3、维护困难:在修改或扩展代码时,可能会意外影响其他部分,因为没有清晰的分隔。随着项目扩展,维护人员可能会对 display 函数的来源感到困惑,增加了调试和代码理解的难度。
4、全局作用域污染:所有定义都在全局作用域中,增加了命名冲突的风险,尤其是在与第三方库交互时。虽然 printSetting 和 printUserSetting 函数存在,但是由于同名全局变量的定义,可能导致意外的行为。例如,可能会误用 setting 变量,造成混淆和难以追踪的错误。

根据上面的案例及总结。我们可以了解到为什么要使用命名空间。
使用命名空间可以有效地解决这些问题,提升代码的组织性和可维护性。

记录的同时,欢迎大家一起补充学习!

相关文章:

cpp中的namespace详解

namespace的作用主要是为了避免名字冲突和组织代码。 命名空间在C中是一个非常重要的特性&#xff0c;它帮助开发者更好地管理代码和避免潜在的冲突。 具体来说&#xff0c;它有以下几个主要用途 避免名字冲突 在大型项目中可能会有很多个类、函数或变量使用相同的名称。使用…...

request库的使用 | get请求

requests 库的 get 方法用于发送 HTTP GET 请求。GET 请求通常用于请求服务器发送数据。 1、导入 requests 库&#xff1a; import requests 2、发送 GET 请求&#xff1a; 使用 requests.get() 方法发送请求。 response requests.get(urlhttp://www.jd.com) 3、查看响应…...

理想低通信道和理想带通信道的区别

一、定义与特性 理想低通信道&#xff08;可通过<MAX&#xff09;&#xff1a; 定义&#xff1a;理想低通信道允许信号的所有低频分量&#xff0c;只要其频率不超过某个上限值&#xff0c;都能够不失真地通过此信道。而频率超过该上限值的所有高频分量都不能通过该信道。特…...

LAMP架构搭建

目录 LAMP架构搭建 编译安装Apache httpd服务 1、需要的安装包 2、关闭防火墙和核心防护 3、安装环境依赖包 4、配置软件模块 5、编译及安装 6、优化配置文件路径&#xff08;可不做&#xff09; 7、添加httpd系统服务 8、修改httpd 服务配置文件 9、浏览器访问验证…...

RT-DETR

SSE represents the single-scale Transformer encoder&#xff0c;CSF represents cross-scale fusion. AIFI and CCFF are the two modules designed into 作者的 hybrid encoder 截止到发文时间的issue数&#xff0c;多吓人呐&#xff0c;不建议复现...

【算法——KMP】

1理解next数组定义&#xff1a;最长相等前后缀&#xff08;不含当前字符并且不能是整体&#xff09; 算法讲解100【扩展】 KMP算法原理和代码详解_哔哩哔哩_bilibili next数组的值&#xff1a;假设这个i出现了不匹配就从next[i]的位置开始在再匹配 2next数组生成 看一下是怎…...

视频监控相关笔记

一、QT 之 QTreeWidget 树形控件 Qt编程指南&#xff0c;Qt新手教程&#xff0c;Qt Programming Guide 一个树形结构的节点中的图表文本 、附带数据的添加&#xff1a; QTreeWidgetItem* TourTreeWnd::InsertNode(NetNodeInfo node, QTreeWidgetItem* parent_item) { // …...

React 中,构建组件的方式

1. 函数组件&#xff08;Function Components&#xff09; 函数组件是最简单的组件形式&#xff0c;通常用于展示性的组件&#xff0c;不涉及复杂的生命周期方法。 import React from react;function Welcome(props) {return <h1>Hello, {props.name}</h1>; }exp…...

Android开发高频面试题之——Android篇

Android开发高频面试题之——Android篇 Android开发高频面试题之——Java基础篇 Android开发高频面试题之——Kotlin基础篇 Android开发高频面试题之——Android基础篇 1. Activity启动模式 standard 标准模式,每次都是新建Activity实例。singleTop 栈顶复用。如果要启动的A…...

禁用拷贝构造函数和赋值构造函数

在C中&#xff0c;禁用拷贝构造函数和拷贝赋值操作符的方式通常是为了防止类的对象被意外复制&#xff0c;这对于那些管理独占资源或不应被复制的对象尤为重要。 class LatActiveControlState : public LatState { public:LatActiveControlState() : LatState(LatS_ActiveCont…...

OneDrive for Business with Office Online 部署方案

目录 前言 部署准备 需求分析 用户需求 技术需求 环境准备 硬件要求 软件要求 许可计划 OneDrive for Business 部署 前期准备 域名配置 Azure AD 配置 安装与配置 安装 OneDrive 同步客户端 配置 OneDrive 组策略 数据迁移 Office Online 部署 前期准备 安…...

win10 win11 设置文件权限以解决Onedrive不能同步问题

初级代码游戏的专栏介绍与文章目录-CSDN博客 我的github&#xff1a;codetoys&#xff0c;所有代码都将会位于ctfc库中。已经放入库中我会指出在库中的位置。 这些代码大部分以Linux为目标但部分代码是纯C的&#xff0c;可以在任何平台上使用。 源码指引&#xff1a;github源…...

Unity DOTS系列之IJobChunk来迭代处理数据

最近DOTS发布了正式的版本, 我们来分享一下System中如何在System中使用IJobChunk来迭代处理World中的数据&#xff0c;方便大家上手学习掌握Unity DOTS开发。 再回顾一次基于ArcheType Chunk内存管理 我们先再次回顾以下基于ArcheType的Chunk内存管理。每一类Entity都是由一些…...

哈希——哈希表

回顾/本期梗概 上期我们学习了哈希——字符串哈希&#xff08;空降链接&#xff09;&#xff0c;本期我们将学习哈希中的哈希表。 1、哈希表原理 &#xff08;1&#xff09;使用数组下标直接标记元素 哈希表&#xff08;也叫数列表&#xff09;&#xff1a;是一种高效的、通过把…...

简单了解 JVM

目录 ♫什么是JVM ♫JVM的运行流程 ♫JVM运行时数据区 ♪虚拟机栈 ♪本地方法栈 ♪堆 ♪程序计数器 ♪方法区/元数据区 ♫类加载的过程 ♫双亲委派模型 ♫垃圾回收机制 ♫什么是JVM JVM 是 Java Virtual Machine 的简称&#xff0c;意为 Java虚拟机。 虚拟机是指通过软件模…...

已经30岁了,想转行从头开始现实吗?什么样的工作算好工作?

我是29岁那年&#xff0c;完成从转行裸辞副业的职业转型。 如果你把职业生涯看成是从现在开始30岁&#xff0c;到你退休那年&#xff0c;中间这么漫长的30年&#xff0c;那么30岁转行完全来得及&#xff1b; 如果你觉得必须在什么年纪&#xff0c;什么时间内必须完成赚到几十…...

快速理解docker(一)docker 简介

在当今快速迭代的软件开发环境中&#xff0c;如何高效地部署、管理和扩展应用程序成为了开发者们面临的重大挑战。Docker&#xff0c;作为一款开源的容器化平台&#xff0c;凭借其轻量级、可移植性和易于部署的特性&#xff0c;迅速成为了解决这些挑战的热门选择。本文将带您走…...

RHCS认证-Linux(RHel9)-Ansible

文章目录 一、ansible 简介二 、ansible部署三、ansible服务端测试四 、ansible 清单inventory五、Ad-hot 点对点模式六、YAML语言模式七、RHCS-Ansible附&#xff1a;安装CentOS-Stream 9系统7.1 ansible 执行过程7.2 安装ansible&#xff0c;ansible-navigator7.2 部署ansibl…...

【Python】Spyder:科学 Python 开发环境

在数据科学和科学计算领域&#xff0c;Python 已经成为了一个不可或缺的工具。为了提高开发效率和改善编程体验&#xff0c;一个功能强大且用户友好的开发环境是必需的。Spyder&#xff08;Scientific Python Development Environment&#xff09;正是这样一个为科学计算和数据…...

SpringBootWeb响应

2. 响应 前面我们学习过HTTL协议的交互方式&#xff1a;请求响应模式&#xff08;有请求就有响应&#xff09; 那么Controller程序呢&#xff0c;除了接收请求外&#xff0c;还可以进行响应。 2.1 ResponseBody 在我们前面所编写的controller方法中&#xff0c;都已经设置了…...

关于nvm与node.js

1 安装nvm 安装过程中手动修改 nvm的安装路径&#xff0c; 以及修改 通过nvm安装node后正在使用的node的存放目录【这句话可能难以理解&#xff0c;但接着往下看你就了然了】 2 修改nvm中settings.txt文件配置 nvm安装成功后&#xff0c;通常在该文件中会出现以下配置&…...

AI,如何重构理解、匹配与决策?

AI 时代&#xff0c;我们如何理解消费&#xff1f; 作者&#xff5c;王彬 封面&#xff5c;Unplash 人们通过信息理解世界。 曾几何时&#xff0c;PC 与移动互联网重塑了人们的购物路径&#xff1a;信息变得唾手可得&#xff0c;商品决策变得高度依赖内容。 但 AI 时代的来…...

Spring是如何解决Bean的循环依赖:三级缓存机制

1、什么是 Bean 的循环依赖 在 Spring框架中,Bean 的循环依赖是指多个 Bean 之间‌互相持有对方引用‌,形成闭环依赖关系的现象。 多个 Bean 的依赖关系构成环形链路,例如: 双向依赖:Bean A 依赖 Bean B,同时 Bean B 也依赖 Bean A(A↔B)。链条循环: Bean A → Bean…...

免费数学几何作图web平台

光锐软件免费数学工具&#xff0c;maths,数学制图&#xff0c;数学作图&#xff0c;几何作图&#xff0c;几何&#xff0c;AR开发,AR教育,增强现实,软件公司,XR,MR,VR,虚拟仿真,虚拟现实,混合现实,教育科技产品,职业模拟培训,高保真VR场景,结构互动课件,元宇宙http://xaglare.c…...

代码规范和架构【立芯理论一】(2025.06.08)

1、代码规范的目标 代码简洁精炼、美观&#xff0c;可持续性好高效率高复用&#xff0c;可移植性好高内聚&#xff0c;低耦合没有冗余规范性&#xff0c;代码有规可循&#xff0c;可以看出自己当时的思考过程特殊排版&#xff0c;特殊语法&#xff0c;特殊指令&#xff0c;必须…...

jmeter聚合报告中参数详解

sample、average、min、max、90%line、95%line,99%line、Error错误率、吞吐量Thoughput、KB/sec每秒传输的数据量 sample&#xff08;样本数&#xff09; 表示测试中发送的请求数量&#xff0c;即测试执行了多少次请求。 单位&#xff0c;以个或者次数表示。 示例&#xff1a;…...

从 GreenPlum 到镜舟数据库:杭银消费金融湖仓一体转型实践

作者&#xff1a;吴岐诗&#xff0c;杭银消费金融大数据应用开发工程师 本文整理自杭银消费金融大数据应用开发工程师在StarRocks Summit Asia 2024的分享 引言&#xff1a;融合数据湖与数仓的创新之路 在数字金融时代&#xff0c;数据已成为金融机构的核心竞争力。杭银消费金…...

Oracle11g安装包

Oracle 11g安装包 适用于windows系统&#xff0c;64位 下载路径 oracle 11g 安装包...

git: early EOF

macOS报错&#xff1a; Initialized empty Git repository in /usr/local/Homebrew/Library/Taps/homebrew/homebrew-core/.git/ remote: Enumerating objects: 2691797, done. remote: Counting objects: 100% (1760/1760), done. remote: Compressing objects: 100% (636/636…...

电脑桌面太单调,用Python写一个桌面小宠物应用。

下面是一个使用Python创建的简单桌面小宠物应用。这个小宠物会在桌面上游荡&#xff0c;可以响应鼠标点击&#xff0c;并且有简单的动画效果。 import tkinter as tk import random import time from PIL import Image, ImageTk import os import sysclass DesktopPet:def __i…...