C++ 命名空间|缺省参数|函数重载

2024-06-11 05:12

本文主要是介绍C++ 命名空间|缺省参数|函数重载,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、命名空间

1.什么是命名空间

命名空间(namespace)是C++中的一种机制,用来解决不同代码库之间的命名冲突问题

先来看一个例子:

#include <iostream>void print() 
{std::cout << "Hello from print()" << std::endl;
}// 假设这里有另一个库也定义了print函数
void print() 
{std::cout << "Hello from another print()" << std::endl;
}int main() 
{print();  // 编译错误,函数名冲突return 0;
}

这是在写大型项目时经常容易遇到的事情,假如这两个库分别由两个不同的人编写,可能这两个库的代码分别能在各自的环境中正常运行,但当代码提交到一起时,就会出现命名冲突问题问题。

为了解决这种问题,C++提出了命名空间的概念,就是通过把全局范围内的变量、函数和类等放在一个逻辑命名空间内,避免名字重叠。 就像这样:

#include <iostream>namespace FirstLibrary 
{void print() {std::cout << "Hello from FirstLibrary::print()" << std::endl;}
}namespace SecondLibrary 
{void print() {std::cout << "Hello from SecondLibrary::print()" << std::endl;}
}int main() 
{FirstLibrary::print();  // 正常调用SecondLibrary::print(); // 正常调用return 0;
}

命名空间包含的成员可以是变量、函数、类和结构体、枚举、类型定义....你还可以套娃在命名空间空间中嵌套命名空间(很少这样做),例如:

namespace MyLibrary
{int a = 10;double b = 1.2;int Add(int left, int right){return left + right;}class MyClass {public:void display() {// 方法实现}};struct Poiont{int x;int y;};enum MyEnum {VALUE1,VALUE2};typedef int MyInt;namespace MyLibrary_2{int a = 0;}//... ...
}

2.如何访问命名空间的成员

2.1作用域展开符

即 直接在要访问的成员名称前加`::`。

`::`是作用域限定符,不作用域限定符则无法访问到该命名空间的成员,因为编译器默认只在全局范围中查找。

int main() 
{// 访问变量std::cout << MyLibrary::a << std::endl;// 调用函数MyLibrary::Add(1,2);// 使用类MyLibrary::MyClass obj;obj.display();// 使用结构体MyLibrary::Poiont  s = { 1, 9 };std::cout << "Point: x = " << s.x << ", y = " << s.y << std::endl;// 使用枚举MyLibrary::MyEnum e = MyLibrary::VALUE1;std::cout << "MyEnum value: " << e << std::endl;// 使用类型别名MyLibrary::MyInt i = 100;std::cout << "MyInt: " << i << std::endl;// 调用嵌套命名空间的函数std::cout << MyLibrary::MyLibrary_2::a << std::endl;return 0;
}

 

2.2命名空间展开

即 使用`using namespace`指令,命名空间展开可以分为全展开和半展开。

全展开:使用using namespace

using namespace MyLibrary;

使用这条语句之后,MyLibrary中的所有成员均可直接访问,即不用在成员前加`::`

#include<iostream>
using namespace std;int main()
{cout << "using namespace std" << endl;return 0;
}

如上面这段代码,使用 using namespace std 将C++标准命名空间展开后,就可以直接访问std里面的cout和endl了

半展开:使用using

using MyLibrary::a;

使用`using`引入特定的成员,这样在程序中即可直接访问该成员

2.3总结

这三种方式各有优劣,直接使用作用域限定符最为清晰明确,但较为冗长;

`using` 声明适合局部引入特定成员;

`using namespace` 指令则最简便,但可能引入命名冲突风险

3.注意事项

  1. 命名空间会自动合并,这意味着可以定义多个同名的命名空间。
  2. 命名空间只能在全局定义!!

  3. `using namespace`  不等于取消命名空间, 影响的是代码编译时候查找该变量的规则(即在指定命名空间和全局变量中寻找), 比如在使用`using namespace std`后, 依然可以使用 `std :: cout` 来使用 `cout`

二、缺省参数

C++ 的缺省参数(也叫默认参数)是指在函数声明或定义时,给某些参数提供一个默认值。这样在调用函数时,如果不传递这些参数,函数会自动使用默认值。比如下面这段代码:

#include<iostream>
using namespace std;// a = 0 b = 1 c = 2即为默认值,函数传参时可以选择不传值
void Func(int a = 0, int b = 1, int c = 2) 
{cout << "a = " << a << endl;cout << "b = " << b << endl;cout << "c = " << c << endl;
}int main()
{cout << "Func(1, 2)\n";Func(1, 2);cout << endl;cout << "Func()\n";Func();return 0;
}

 

1.缺省参数分类

1.1全缺省参数

void Func(int a = 10, int b = 20, int c = 30)

1.2半缺省参数

void Func(int a, int b = 10, int c = 20)

2.使用缺省参数的注意事项

2.1申明缺省参数时从右到左

缺省参数必须从右到左依次出现,例如,不能先给 a 指定默认值而不给 b 默认值。

// 错误
void func(int a = 5, int b);// 正确
void func(int a, int b = 10);

2.2传递参数时从左到右

不存在不连续的缺省值,即 例如在传参时不能省略第一个参数,而给出第二个参数

#include<iostream>
using namespace std;void Func(int a = 0, int b  = 1, int c = 2) // a = 0 是舔狗 没人的时候它就上
{cout << "a = " << a << endl;cout << "b = " << b << endl;cout << "c = " << c << endl;
}int main()
{Func(, 2, 3); //这样是不正确的!return 0;
}

 

2.3如果申明和定义分离,则只在声明中写缺省参数

如果函数在头文件中声明并在源文件中定义,则缺省参数不能在函数声明和定义中同时出现,所以在函数声明的时候给缺省参数, 定义的时候不给缺省参数。

因为如果声明与定义位置同时出现,恰巧两个位置提供的值不同,编译器就无法确定到底该用哪个缺省值。

2.4缺省值必须是常量或者全局变量

这一点显然,给出的缺省值一定是一个固定的值,而不是一个变化的值,否则这个缺省值将毫无意义

三、函数重载

1.什么是函数重载

C++ 的函数重载是指在同一个作用域中,可以定义多个同名但参数不同的函数。编译器会根据调用时的实参类型和数量来决定调用哪个函数。这使得程序更具灵活性和可读性。

或许你已经发现了,在C++中,`std::cout` 不需要像 `printf()` 一样给定参数类型,它能自动判断参数类型输出,这其实就是一种函数重载。

在C语言中,由于不存在函数重载,我们只能通过区分函数名称来处理不同类型的数据

void printInt(int i) 
{printf("%d", i);
}void printDouble(double f) 
{printf("%f", f);
}void printString(char s[]) 
{printf("%s", s);
}

有了函数重载之后: 

void print(int i) 
{cout << "整数: " << i << endl;
}void print(double f) 
{cout << "浮点数: " << f << endl;
}void print(string s) 
{cout << "字符串: " << s << endl;
}int main() {print(10);          // 调用 print(int)print(3.14);        // 调用 print(double)print("Hello");     // 调用 print(string)return 0;
}

我们只需定义一个函数名 print,根据不同的参数类型调用不同的实现,代码更简洁易读。 

2.构成函数重载的关键点

  1. 函数名相同:所有重载函数必须有相同的名字。        

    但注意,函数重载强调在同一个作用域中,所以两个不同命名空间的同名函数不构成函数重载

  2. 参数不同:重载函数的参数类型、个数或顺序至少有一个不同
  3. 返回类型:返回类型可以不同,但仅靠返回类型不同不能构成重载。

 

这篇关于C++ 命名空间|缺省参数|函数重载的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1050281

相关文章

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

Three.js构建一个 3D 商品展示空间完整实战项目

《Three.js构建一个3D商品展示空间完整实战项目》Three.js是一个强大的JavaScript库,专用于在Web浏览器中创建3D图形,:本文主要介绍Three.js构建一个3D商品展... 目录引言项目核心技术1. 项目架构与资源组织2. 多模型切换、交互热点绑定3. 移动端适配与帧率优化4. 可

SpringBoot 获取请求参数的常用注解及用法

《SpringBoot获取请求参数的常用注解及用法》SpringBoot通过@RequestParam、@PathVariable等注解支持从HTTP请求中获取参数,涵盖查询、路径、请求体、头、C... 目录SpringBoot 提供了多种注解来方便地从 HTTP 请求中获取参数以下是主要的注解及其用法:1

HTTP 与 SpringBoot 参数提交与接收协议方式

《HTTP与SpringBoot参数提交与接收协议方式》HTTP参数提交方式包括URL查询、表单、JSON/XML、路径变量、头部、Cookie、GraphQL、WebSocket和SSE,依据... 目录HTTP 协议支持多种参数提交方式,主要取决于请求方法(Method)和内容类型(Content-Ty

GO语言中函数命名返回值的使用

《GO语言中函数命名返回值的使用》在Go语言中,函数可以为其返回值指定名称,这被称为命名返回值或命名返回参数,这种特性可以使代码更清晰,特别是在返回多个值时,感兴趣的可以了解一下... 目录基本语法函数命名返回特点代码示例命名特点基本语法func functionName(parameters) (nam

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

Python Counter 函数使用案例

《PythonCounter函数使用案例》Counter是collections模块中的一个类,专门用于对可迭代对象中的元素进行计数,接下来通过本文给大家介绍PythonCounter函数使用案例... 目录一、Counter函数概述二、基本使用案例(一)列表元素计数(二)字符串字符计数(三)元组计数三、C

python中的显式声明类型参数使用方式

《python中的显式声明类型参数使用方式》文章探讨了Python3.10+版本中类型注解的使用,指出FastAPI官方示例强调显式声明参数类型,通过|操作符替代Union/Optional,可提升代... 目录背景python函数显式声明的类型汇总基本类型集合类型Optional and Union(py

C++ STL-string类底层实现过程

《C++STL-string类底层实现过程》本文实现了一个简易的string类,涵盖动态数组存储、深拷贝机制、迭代器支持、容量调整、字符串修改、运算符重载等功能,模拟标准string核心特性,重点强... 目录实现框架一、默认成员函数1.默认构造函数2.构造函数3.拷贝构造函数(重点)4.赋值运算符重载函数

Python中的filter() 函数的工作原理及应用技巧

《Python中的filter()函数的工作原理及应用技巧》Python的filter()函数用于筛选序列元素,返回迭代器,适合函数式编程,相比列表推导式,内存更优,尤其适用于大数据集,结合lamb... 目录前言一、基本概念基本语法二、使用方式1. 使用 lambda 函数2. 使用普通函数3. 使用 N