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

函数重载(Function Overloading)


1. 函数重载的核心概念

函数重载允许在 同一作用域内定义多个同名函数,但它们的 参数列表(参数类型、顺序或数量)必须不同。编译器在编译时根据 调用时的实参类型和数量 静态选择最匹配的函数版本。


2. 源码示例:基础函数重载

示例 1:不同参数类型的重载
#include <iostream>
using namespace std;// 重载函数:参数类型不同
void print(int a) {cout << "整数: " << a << endl;
}void print(double a) {cout << "浮点数: " << a << endl;
}void print(const char* a) {cout << "字符串: " << a << endl;
}int main() {print(10);        // 调用 void print(int)print(3.14);      // 调用 void print(double)print("Hello");   // 调用 void print(const char*)return 0;
}

输出:

整数: 10
浮点数: 3.14
字符串: Hello
关键点:
  • 编译器根据实参类型选择函数,例如 10int,直接匹配 print(int)
  • 3.14 默认是 double(不是 float),因此匹配 print(double)
  • 字符串字面量 "Hello" 的类型是 const char*,匹配第三个函数。

3. 函数重载的底层实现

C++ 编译器通过 名称修饰(Name Mangling) 为每个重载函数生成唯一的符号名。例如:

  • void print(int)_Z5printi
  • void print(double)_Z5printd
  • void print(const char*)_Z5printPKc
查看符号名的方法(Linux/g++)
  1. 编译代码并生成目标文件:
    g++ -c overload.cpp -o overload.o
    
  2. 使用 nm 命令查看符号表:
    nm overload.o
    
    输出类似:
    00000000 T _Z5printi
    0000000a T _Z5printd
    00000014 T _Z5printPKc
    
    这里 T 表示符号在代码段(Text Section),后面是修饰后的函数名。

4. 示例 2:参数数量和顺序不同

代码示例
#include <iostream>
using namespace std;// 参数数量不同
int add(int a, int b) {return a + b;
}int add(int a, int b, int c) {return a + b + c;
}// 参数顺序不同
void process(int a, double b) {cout << "int, double: " << a << ", " << b << endl;
}void process(double a, int b) {cout << "double, int: " << a << ", " << b << endl;
}int main() {cout << add(1, 2) << endl;        // 调用 add(int, int)cout << add(1, 2, 3) << endl;     // 调用 add(int, int, int)process(10, 3.14);                // 调用 process(int, double)process(3.14, 10);               // 调用 process(double, int)return 0;
}

输出:

3
6
int, double: 10, 3.14
double, int: 3.14, 10
关键点:
  • 参数数量不同(如 add 的两个版本)。
  • 参数顺序不同(如 process 的两个版本),编译器根据实参顺序匹配。

5. 函数重载的歧义问题

示例 1:默认参数导致歧义
void func(int a, int b = 0) {cout << "func(int, int)" << endl;
}void func(int a) {cout << "func(int)" << endl;
}int main() {func(10);  // ❌ 编译错误:无法确定调用哪个函数return 0;
}

问题分析:

  • 调用 func(10) 可以匹配 func(int),也可以匹配 func(int, int)(第二个参数使用默认值 0)。
  • 编译器无法确定选择哪个函数,因此报错。

示例 2:类型转换歧义
void print(long a) {cout << "long: " << a << endl;
}void print(double a) {cout << "double: " << a << endl;
}int main() {print(10);  // ❌ 编译错误:无法确定调用哪个函数return 0;
}

问题分析:

  • 10int 类型,可以隐式转换为 longdouble
  • 编译器无法确定哪种转换更优,因此报错。

6. 重载函数的优先级规则

当存在多个可能的匹配时,编译器按以下优先级选择:

  1. 精确匹配(参数类型完全一致)。
  2. 提升转换(如 charintfloatdouble)。
  3. 标准转换(如 intlongdoubleint)。
  4. 用户自定义转换(如通过构造函数或转换运算符)。
示例:优先级选择
void print(int a) {cout << "int: " << a << endl;
}void print(double a) {cout << "double: " << a << endl;
}int main() {short s = 10;print(s);  // 调用 print(int)(short 提升为 int)print(10L); // 调用 print(double)(long 转换为 double)return 0;
}

输出:

int: 10
double: 10

7. 类成员函数的重载

函数重载不仅适用于全局函数,也适用于类的成员函数。

示例:类中的重载
#include <iostream>
using namespace std;class Calculator {
public:int add(int a, int b) {return a + b;}double add(double a, double b) {return a + b;}string add(const string& a, const string& b) {return a + b;}
};int main() {Calculator calc;cout << calc.add(1, 2) << endl;           // 调用 int add(int, int)cout << calc.add(1.5, 2.5) << endl;       // 调用 double add(double, double)cout << calc.add("Hello, ", "World!") << endl; // 调用 string add(const string&, const string&)return 0;
}

输出:

3
4
Hello, World!

8. 函数重载 vs. 模板

函数重载需要为每个类型手动编写函数,而模板可以自动生成代码。

函数重载实现
int max(int a, int b) { return (a > b) ? a : b; }
double max(double a, double b) { return (a > b) ? a : b; }
模板实现
template <typename T>
T max(T a, T b) {return (a > b) ? a : b;
}

区别:

  • 重载需要显式定义每个类型。
  • 模板通过泛型编程自动适配类型,但可能无法处理某些特殊逻辑。

9. 总结

  • 函数重载是静态多态的核心机制,通过编译时绑定实现高效代码。
  • 核心规则:同名函数参数列表不同。
  • 编译器通过 名称修饰(Name Mangling) 生成唯一符号。
  • 注意避免歧义(如默认参数、隐式转换冲突)。

多选题

题目 1:函数重载的核心规则

以下哪些情况可以构成合法的函数重载?

A. 两个函数的参数类型不同:void func(int);void func(double);
B. 两个函数的参数数量不同:void func(int);void func(int, int);
C. 两个函数的返回值类型不同:int func(int);double func(int);
D. 两个函数的参数顺序不同:void func(int, double);void func(double, int);


题目 2:类型转换与重载歧义

以下代码是否会编译失败?为什么?

void print(long a) { /* ... */ }
void print(double a) { /* ... */ }int main() {print(10);  // 调用哪个函数?return 0;
}

A. 编译成功,调用 print(long),因为 intlong 是标准转换
B. 编译成功,调用 print(double),因为 intdouble 是提升转换
C. 编译失败,因为 int 可以隐式转换为 longdouble,导致歧义
D. 编译成功,因为 intdouble 的转换优先级更高


题目 3:默认参数与重载冲突

以下代码是否会编译失败?

void process(int a, int b = 0) { /* ... */ }
void process(int a) { /* ... */ }int main() {process(10);  // 调用哪个函数?return 0;
}

A. 编译成功,调用 process(int)
B. 编译成功,调用 process(int, int)
C. 编译失败,因为默认参数导致歧义
D. 编译成功,因为默认参数的优先级更低


题目 4:模板函数与重载的优先级

以下代码的输出是什么?

#include <iostream>
using namespace std;template <typename T>
void print(T a) { cout << "模板函数: " << a << endl; }void print(int a) { cout << "重载函数: " << a << endl; }int main() {print(10);     // 调用哪个函数?print(10.5);   // 调用哪个函数?return 0;
}

A. 两次都调用模板函数
B. 两次都调用重载函数
C. print(10) 调用重载函数,print(10.5) 调用模板函数
D. print(10) 调用模板函数,print(10.5) 调用重载函数


题目 5:作用域与函数重载

以下代码的输出是什么?

#include <iostream>
using namespace std;class Base {
public:void func(int a) { cout << "Base::func(int)" << endl; }
};class Derived : public Base {
public:void func(double a) { cout << "Derived::func(double)" << endl; }
};int main() {Derived obj;obj.func(10);     // 调用哪个函数?obj.func(10.5);   // 调用哪个函数?return 0;
}

A. 两次都调用 Derived::func(double)
B. func(10) 调用 Base::func(int)func(10.5) 调用 Derived::func(double)
C. 编译成功,Derived 隐藏了 Base::func(int)
D. 两次都调用 Base::func(int)



答案与解析

题目 1:函数重载的核心规则

答案:A、B、D
解析

  • A 正确:参数类型不同是合法的重载。
  • B 正确:参数数量不同是合法的重载。
  • C 错误:仅返回值不同不足以构成重载。
  • D 正确:参数顺序不同是合法的重载。

题目 2:类型转换与重载歧义

答案:C
解析

  • int 可以隐式转换为 long(标准转换)或 double(提升转换),但两者优先级相同,编译器无法确定哪个更优,导致歧义,编译失败。
  • 提升转换(如 intdouble)和标准转换(如 intlong)的优先级在 C++ 中可能因编译器实现不同,但此处两者没有明确的优先级高低。

题目 3:默认参数与重载冲突

答案:C
解析

  • process(10) 可以匹配 process(int)process(int, int)(第二个参数使用默认值),编译器无法确定调用哪个函数,导致编译失败。

题目 4:模板函数与重载的优先级

答案:C
解析

  • 当模板函数和非模板重载函数同时匹配时,非模板函数优先级更高。
  • print(10) 精确匹配 void print(int),调用重载函数。
  • print(10.5) 没有重载的 double 版本,调用模板函数。

题目 5:作用域与函数重载

答案:A、C
解析

  • C 正确:派生类中定义的 func(double) 会隐藏基类的 func(int)(即使参数不同)。
  • A 正确:所有调用都会优先查找派生类的作用域,因此 obj.func(10) 会将 int 隐式转换为 double,调用 Derived::func(double)
  • B 错误:基类的 func(int) 被隐藏,无法直接调用。

相关文章:

函数重载(Function Overloading)

1. 函数重载的核心概念 函数重载允许在 同一作用域内定义多个同名函数&#xff0c;但它们的 参数列表&#xff08;参数类型、顺序或数量&#xff09;必须不同。编译器在编译时根据 调用时的实参类型和数量 静态选择最匹配的函数版本。 2. 源码示例&#xff1a;基础函数重载 示…...

CGAL 网格等高线计算

文章目录 一、简介二、实现代码三、实现效果一、简介 这里等高线的计算其实很简单,使用不同高度的水平面与网格进行相交,最后获取不同高度的相交线即可。 二、实现代码 #include <iostream> #include <iterator> #include <map>...

计算机组成与体系结构:缓存(Cache)

目录 为什么需要 Cache&#xff1f; &#x1f9f1; Cache 的分层设计 &#x1f539; Level 1 Cache&#xff08;L1 Cache&#xff09;一级缓存 &#x1f539; Level 2 Cache&#xff08;L2 Cache&#xff09;二级缓存 &#x1f539; Level 3 Cache&#xff08;L3 Cache&am…...

Flutter 在全新 Platform 和 UI 线程合并后,出现了什么大坑和变化?

Flutter 在全新 Platform 和 UI 线程合并后&#xff0c;出现了什么大坑和变化&#xff1f; 在两个月前&#xff0c;我们就聊过 3.29 上《Platform 和 UI 线程合并》的具体原因和实现方式&#xff0c;而事实上 Platform 和 UI 线程合并&#xff0c;确实为后续原生语言和 Dart 的…...

开发 MCP Proxy(代理)也可以用 Solon AI MCP 哟!

MCP 有三种通讯方式&#xff1a; 通道说明备注stdio本地进程内通讯现有sse http远程 http 通讯现有streamable http远程 http 通讯&#xff08;MCP 官方刚通过决定&#xff0c;mcp-java-sdk 还没实现&#xff09; 也可以按两大类分&#xff1a; 本地进程间通讯远程通讯&…...

JetBrains GoLang IDE无限重置试用期,适用最新2025版

注意本文仅用于学习使用&#xff01;&#xff01;&#xff01; 本文在重置2024.3.5版本亲测有效&#xff0c;环境为window(mac下应该也一样奏效) 之前eval-reset插件只能在比较低的版本才能起作用。 总结起来就一句&#xff1a;卸载重装&#xff0c;额外要删掉旧安装文件和注册…...

python中socket(套接字)库详细解析

目录 1. 前言 2. socket 库基础 2.1 什么是 socket&#xff1f; 2.2 socket 的类型 3. 基于 TCP 的 socket 编程 3.1 TCP 服务器端代码示例 3.2 TCP 客户端代码示例 3.3 代码分析 4. 基于 UDP 的 socket 编程 4.1 UDP 服务器端代码示例 4.2 UDP 客户端代码示例 4.3…...

鸿蒙-状态管理V1和V2在ForEach循环渲染的表现

目录 前提遇到的问题换V2呗 状态管理V2已经出来好长时间了&#xff0c;移除GAP说明也有一段时间了&#xff0c;相信有一部分朋友已经开始着手从V1迁移到V2了&#xff0c;应该也踩了不少坑。 下面向大家分享一下我使用状态管理V1和Foreach时遇到的坑&#xff0c;以及状态管理V2在…...

深入了解递归、堆与栈:C#中的内存管理与函数调用

在编程中&#xff0c;理解如何有效地管理内存以及如何控制程序的执行流程是每个开发者必须掌握的基本概念。C#作为一种高级编程语言&#xff0c;其内存管理和函数调用机制包括递归、堆与栈。本文将详细讲解这三者的工作原理、用途以及它们在C#中的实现和应用。 1. 递归 (Recur…...

图论---Prim堆优化(稀疏图)

题目通常会提示数据范围&#xff1a; 若 V ≤ 500&#xff0c;两种方法均可&#xff08;朴素Prim更稳&#xff09;。 若 V ≤ 1e5&#xff0c;必须用优先队列Prim vector 存图。 #include <iostream> #include <vector> #include <queue> #include <…...

stm32之GPIO函数详解和上机实验

目录 1.LED和蜂鸣器1.1 LED1.2 蜂鸣器 2.实验2.1 库函数&#xff1a;RCC和GPIO2.1.1 RCC函数1. RCC_AHBPeriphClockCmd2. RCC_APB2PeriphClockCmd3. RCC_APB1PeriphClockCmd 2.1.2 GPIO函数1. GPIO_DeInit2. GPIO_AFIODeInit3. GPIO_Init4. GPIO_StructInit5. GPIO_ReadInputDa…...

用 PyQt5 和 asyncio 打造接口并发测试 GUI 工具

接口并发测试是测试工程师日常工作中的重要一环&#xff0c;而一个直观的 GUI 工具能有效提升工作效率和体验。本篇文章将带你用 PyQt5 和 asyncio 从零实现一个美观且功能实用的接口并发测试工具。 我们将实现以下功能&#xff1a; 请求方法选择器 添加了一个下拉框 QComboBo…...

OpenHarmony Camera开发指导(四):相机会话管理(ArkTS)

概述 相机在使用预览、拍照、录像、获取元数据等功能前&#xff0c;都需要先创建相机会话。 相机会话Session的功能如下&#xff1a; 配置相机的输入流和输出流。 配置输入流即添加设备输入&#xff0c;通俗来讲即选择某一个摄像头进行拍照录像&#xff1b;配置输出流&#x…...

深入探索RAG(检索增强生成)模型的优化技巧

&#x1f4cc; 友情提示&#xff1a; 本文内容由银河易创AI&#xff08;https://ai.eaigx.com&#xff09;创作平台的gpt-4o-mini模型生成&#xff0c;旨在提供技术参考与灵感启发。文中观点或代码示例需结合实际情况验证&#xff0c;建议读者通过官方文档或实践进一步确认其准…...

Spring boot 中的IOC容器对Bean的管理

Spring Boot 中 IOC 容器对 Bean 的管理&#xff0c;涵盖从容器启动到 Bean 的生命周期管理的全流程。 步骤 1&#xff1a;理解 Spring Boot 的容器启动 Spring Boot 的 IOC 容器基于 ApplicationContext&#xff0c;在应用启动时自动初始化。 入口类&#xff1a;通过 SpringB…...

Qt实战之将自定义插件(minGW)显示到Qt Creator列表的方法

Qt以其强大的跨平台特性和丰富的功能&#xff0c;成为众多开发者构建图形用户界面&#xff08;GUI&#xff09;应用程序的首选框架。而在Qt开发的过程中&#xff0c;自定义插件能够极大地拓展应用程序的功能边界&#xff0c;让开发者实现各种独特的、个性化的交互效果。想象一下…...

【Vue】TypeScript与Vue3集成

个人主页&#xff1a;Guiat 归属专栏&#xff1a;Vue 文章目录 1. 前言2. 环境准备与基础搭建2.1. 安装 Node.js 与 npm/yarn/pnpm2.2. 创建 Vue3 TypeScript 项目2.2.1. 使用 Vue CLI2.2.2. 使用 Vite&#xff08;推荐&#xff09;2.2.3. 目录结构简述 3. Vue3 TS 基础语法整…...

Linux之七大难命令(The Seven Difficult Commands of Linux)

Linux之七大难命令 、背景 作为Linux的初学者&#xff0c;肯定要先掌握高频使用的指令&#xff0c;这样才能让Linux的学习在短时间内事半功倍。但是&#xff0c;有些指令虽然功能强大&#xff0c;但因参数多而让初学者们很害怕&#xff0c;今天介绍Linux中高频使用&#xff0…...

Spring Boot单元测试实战指南:从零到高效测试

在Spring Boot开发中&#xff0c;单元测试是保障代码质量的核心环节。本文将基于实际开发场景&#xff0c;手把手教你如何快速实现分层测试、模拟依赖、编写高效断言&#xff0c;并分享最佳实践&#xff01; 一、5分钟环境搭建 添加依赖 在pom.xml中引入spring-boot-starter-te…...

5.3.1 MvvmLight以及CommunityToolkit.Mvvm介绍

MvvmLight、CommunityToolkit.Mvvm是开源包,他们为实现 MVVM(Model-View-ViewModel)模式提供了一系列实用的特性和工具,能帮助开发者更高效地构建 WPF、UWP、MAUI 等应用程序。 本文介绍如下: 一、使用(旧)的MvvmLight库 其特点如下,要继承的基类是ViewModelBase;且使用…...

Dbeaver 执行 SQL 语句和执行 SQL 脚本的区别

执行 SQL 语句 执行 SQL 语句对应图标&#xff1a; 适用于执行单个 SQL 的情形&#xff0c;默认是在光标处或选中的文本上执行 SQL 查询。 实际上同时选择多个 SQL 并通过该方式去执行也可能成功&#xff0c;只是有失败的风险。因此不建议使用它来同时执行多个 SQL 语句。 情况…...

《Python3网络爬虫开发实战(第二版)》配套案例 spa6

Scrape | Moviehttps://spa6.scrape.center/ 请求影片列表api时&#xff0c;不仅有分页参数&#xff0c;还多了一个token&#xff0c;通过重发请求发现token有时间限制&#xff0c;所以得逆向token的生成代码。 通过xhr断点定位到接口请求位置 刷新页面或者点翻页按钮&#x…...

AWS 中国区 CloudFront SSL 证书到期更换实战指南

适用场景: AWS 中国区(宁夏区域 cn-northwest-1 或北京区域 cn-north-1)CloudFront 分配的 SSL 证书到期后无缝替换,域名主体为 domain.cn。 背景与痛点 当 CloudFront 使用的 SSL 证书即将到期时,需手动替换新证书以避免服务中断。由于 AWS 中国区 不支持 ACM 证书,必须…...

Python基础语法:字面量,注释,关键字,标识符,变量和引用,程序执行的3大流程

目录 字面量&#xff08;数据的类型&#xff09; 字面量的含义 常见字面量类型&#xff08;6种&#xff09; 输出各类字面量&#xff08;print语句&#xff09; 注释&#xff08;单行和多行注释&#xff09; 注释的作用 单行注释和多行注释 单行注释&#xff08;ctrl/&a…...

SPL 量化 获取数据

下载数据 我们将股票数据分享在百度网盘上供下载&#xff0c;每工作日更新。 目前可供下载的数据有 A 股的日 K 线数据、股票代码列表和上市公司的基本面数据 下载链接&#xff1a; 百度网盘 下载数据的文件格式为 btx&#xff0c;是 SPL 的特有二进制格式。 btx 称为集文…...

VMware与Docker:虚拟化技术的双轨演进与融合实践

一、虚拟化的本质与价值重构 虚拟化&#xff08;Virtualization&#xff09;是通过软件抽象层将物理资源转化为可动态分配的虚拟单元&#xff0c;其核心价值在于打破"一机一用"的刚性架构&#xff0c;实现三大突破性转变&#xff1a; 资源解耦&#xff1a;硬件资源…...

3. pandas笔记之:创建

以下是 Pandas 主要数据结构的创建方式整理&#xff0c;涵盖 Series 和 DataFrame 的常见创建方法&#xff1a; 一、Series 创建方式 从列表/数组创建 import pandas as pd import numpy as np# 基础列表 s1 pd.Series([1, 3, 5, np.nan, 6])# 指定索引 s2 pd.Series([10, …...

潞晨科技将暂停DeepSeek API服务,AI大模型技术红利普惠化与市场竞争白热化叠加,内卷恶果,开始显现!

潞晨科技宣布暂停DeepSeek API服务的事件,不仅暴露了AI大模型行业的技术与成本博弈,更折射出国内AI生态中中小企业的生存困境和行业内卷的深层矛盾。这一事件背后,既有企业个体商业模式的局限性,也揭示了整个行业在技术迭代、成本控制和市场策略上的系统性挑战。 一、潞晨科…...

某大型电解铝厂电解系统谐波治理装置改造沃伦森电气

电解铝行业谐波治理解决方案——无源滤波装置优化升级&#xff0c;保障稳定运行 在电解铝生产过程中&#xff0c;谐波污染问题严重影响电网电能质量&#xff0c;甚至可能导致滤波装置损坏&#xff0c;引发群爆事故。河南登封某大型电解铝厂通过无源滤波装置智能化改造&#xff…...

Rust 学习笔记:安装 Rust

Rust 学习笔记&#xff1a;安装 Rust Rust 学习笔记&#xff1a;安装 Rust在 Windows 上安装 Rust命令行创建 Rust 项目在 Mac/Linux 上安装 Rust一些命令升级卸载cargo -hrustc -h 安装 RustRoverrust-analyzer Rust 学习笔记&#xff1a;安装 Rust 在 Windows 上安装 Rust …...