函数重载(Function Overloading)

news2025/4/25 11:04:01

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) 被隐藏,无法直接调用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2342365.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

计算机组成与体系结构:缓存(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 的…

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…

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…

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…

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

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

SPL 量化 获取数据

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

Rust 学习笔记:安装 Rust

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

编译 C++ 报错“找不到 g++ 编译器”的终极解决方案(含 Windows/Linux/macOS)

前言 在使用终端编译 C 程序时&#xff0c;报错&#xff1a; 或类似提示&#xff0c;意味着你的系统尚未正确安装或配置 g 编译器。本篇将从零手把手教你在 Windows / Linux / macOS 下安装并配置 g&#xff0c;适用于新手或 C 入门阶段的你。 什么是 g&#xff1f; g 是 GN…

html单页业务介绍源码

源码介绍 html单页业务介绍源码&#xff0c;源码由HTMLCSSJS组成&#xff0c;记事本打开源码文件可以进行内容文字之类的修改&#xff0c;双击html文件可以本地运行 效果预览 源码免费获取 html单页业务介绍源码

单体OJ项目

单体项目版本、微服务版还需我再钻研钻研。 项目介绍 在系统前台&#xff0c;管理员可以创建、管理题目;用户可以自由搜索题目、阅读题目、编写并提交代码。 在系统后端&#xff0c;能够根据管理员设定的题目测试用例在代码沙箱 中对代码进行编译、运行、判断输出是否正确。 其…

豆包桌面版 1.47.4 可做浏览器,免安装绿色版

自己动手升级更新办法&#xff1a; 下载新版本后安装&#xff0c;把 C:\Users\用户名\AppData\Local\Doubao\Application 文件夹的文件&#xff0c;拷贝替换 DoubaoPortable\App\Doubao 文件夹的文件&#xff0c;就升级成功了。 再把安装的豆包彻底卸载就可以。 桌面版比网页版…

【MySQL】索引失效问题详解

目录 1. 最左前缀原则 2. 条件左边有函数或运算 3. 隐式类型转换 4. LIKE 模糊查询以 % 开头 5、MySQL 优化器选择全表扫描 ⭐对 in 关键字特别说明⭐ &#xff08;1&#xff09;列表太大时&#xff0c;走全表扫描了 &#xff08;2&#xff09;隐式类型转换 &#xff…

优选算法第十讲:字符串

优选算法第十讲&#xff1a;字符串 1.最长公共前缀2.最长回文子串3.二进制求和4.字符串相乘 1.最长公共前缀 2.最长回文子串 3.二进制求和 4.字符串相乘

【扣子Coze 智能体案例四】五行八卦占卜智能体

目录 一、意图识别 二、时间格式转换 三、八字转换 四、八字提取 五、八字提取2 六、数据汇总 七、统计五行占比 八、雷达图生成 九、表格生成 十、AI占卜 十一、结束节点 一、意图识别 用户输入的信息包含各种时间格式的年月日时 用户输入的信息包含天干地支八字…