我的编程空间,编程开发者的网络收藏夹
学习永远不晚

如何使用C++获取指定的重载函数地址

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

如何使用C++获取指定的重载函数地址

刚刚看到一篇博客,说 std::bind 无法绑定正确的重载函数。这里的问题并不是 std::bind 能力不足,而是将函数名传递给 std::bind 时编译器无法取到这个函数的地址(也就是符号,编译器会先解析成符号,链接器再替换为地址),因为有多个重载函数都是这个名字。核心问题是无法通过函数名取到想要的重载函数地址。就像下面的代码无法编译通过:

#include <iostream>
void f()
{
    std::cout << "f 1" << std::endl;
}

void f(int x)
{
    std::cout << "f 2 " << x << std::endl;
}

int main()
{
    auto p = &f;
}

编译错误:

/home/abc/cpplearn/overload_func.cpp: In function ‘int main()’:
/home/abc/cpplearn/overload_func.cpp:15:15: error: unable to deduce ‘auto’ from ‘& f’
   15 |     auto p = &f;
      |               ^
/home/abc/cpplearn/overload_func.cpp:15:15: note:   couldn’t deduce template parameter ‘auto’

有没有什么比较完美的解决办法呢?我觉得一定有,因为 C 语言没有函数重载,函数地址作为实参也是常规操作。相比之下,C++ 引入了函数重载,却无法取到函数地址,这就很尴尬。C++ 设计者肯定也想到了这个问题。

于是查阅了 cppreference.com,看到了 Address of an overloaded function。函数名的重载解析除了发生在函数调用的时候,也会发生在以下 7 种语境:

#ContextTarget
1initializer in a declaration of an object or referencethe object or reference being initialized
2on the right-hand-side of an assignment expressionthe left-hand side of the assignment
3as a function call argumentthe function parameter
4as a user-defined operator argumentthe operator parameter
5the return statementthe return type of a function
6explicit cast or static_cast argumentthe target type of a cast
7non-type template argumentthe type of the template parameter

当函数名存在于这 7 种语境时,会发生重载解析,并且会选择与 Target 类型匹配的那个重载函数。这里就不一一考察这 7 种语境了,有兴趣可以自己查阅 cppreference.com。这里重点考察第 3 种和第 6 种。

先看第 3 种语境。当函数名作为函数调用的实参时,重载解析会选择和形参类型相匹配的版本。也就是说,下面的代码会如期运行:

#include <iostream>
void f()
{
    std::cout << "f 1" << std::endl;
}

void f(int x)
{
    std::cout << "f 2 " << x << std::endl;
}

void call(void p(int)) {
    p(1);
}

int main()
{
    call(f);
}

这段代码输出:

f 2 1

回到最初的问题,std::bind 也是函数,为什么无法正常编译呢?直接分析下面代码的编译错误信息:

#include <iostream>
#include <functional>

void f()
{
    std::cout << "f 1" << std::endl;
}

void f(int x)
{
    std::cout << "f 2 " << x << std::endl;
}

int main()
{
    auto new_func = std::bind(f, std::placeholders::_1);
    new_func(66);
}

编译错误:

/home/abc/cpplearn/overload_func.cpp: In function ‘int main()’:
/home/abc/cpplearn/overload_func.cpp:16:30: error: no matching function for call to ‘bind(<unresolved overloaded function type>, const std::_Placeholder<1>&)’
   16 |     auto new_func = std::bind(f, std::placeholders::_1);
      |                     ~~~~~~~~~^~~~~~~~~~~~~~~~~~~~~~~~~~

可以看到,std::bind 准确地说是一个函数模板。它要根据其参数进行模板实参推导,再替换模板形参进行实例化(Instantiation),产生和普通函数类似的汇编代码。std::bind 进行实例化的时候,函数 f 还没有进行重载解析,其类型为<unresolved overloaded function type>。std::bind 无法进行实例化。怎样修改可以解决这个问题呢?

可以利用第 6 个语境,也就是显示转换或 static_cast。重载解析会选择与它们的目标类型相匹配的版本。下面的代码会如期运行:

#include <iostream>
#include <functional>

void f()
{
    std::cout << "f 1" << std::endl;
}

void f(int x)
{
    std::cout << "f 2 " << x << std::endl;
}

int main()
{
    auto new_func = std::bind((void(*)(int))f, std::placeholders::_1);
    new_func(66);
}

这段代码输出:

f 2 66

还有一种更加巧妙的办法,依然是利用第 3 种语境。既然 std::bind 的第一个模板实参的推导,和 f 的重载解析相矛盾。为什么不直接解决这个矛盾,将第一个模板实参改为显示指定?来看下面的代码:

#include <iostream>
#include <functional>

void f()
{
    std::cout << "f 1" << std::endl;
}

void f(int x)
{
    std::cout << "f 2 " << x << std::endl;
}

int main()
{
    auto new_func = std::bind<void(int)>(f, std::placeholders::_1);
    new_func(66);
}

这段代码如期输出:

f 2 66

总结

到此这篇关于如何使用C++获取指定的重载函数地址的文章就介绍到这了,更多相关C++获取重载函数地址内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

如何使用C++获取指定的重载函数地址

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

如何使用 C++ 函数指针重载和泛型编程?

c++++ 函数指针重载通过指定不同函数签名实现指向具有相同名称但不同参数或返回值的多函数指针。泛型编程使用模板创建适用于不同类型数据的函数和数据结构,使代码可重用。使用函数指针重载需要为每种类型编写单独的函数,而泛型编程则使用通用函数处理
如何使用 C++ 函数指针重载和泛型编程?
2024-04-17

如何使用 Golang 获取我的 wifi IP 地址?

php小编小新在这里为大家介绍如何使用Golang获取自己的wifi IP地址。Golang是一种高效且简洁的编程语言,它提供了很多方便的方法来处理网络相关的任务。要获取wifi IP地址,我们可以使用Golang的net包和os/exec
如何使用 Golang 获取我的 wifi IP 地址?
2024-02-09

C++ 函数重载如何与虚函数结合使用?

函数重载和虚函数可以结合使用,允许子类在不修改父类行为的情况下,以不同方式实现相同操作的不同方面。通过在父类中声明虚函数,并在子类中重载它们,我们可以实现动态多态,允许通过父类引用调用不同子类的特定函数。C++ 函数重载与虚函数的结合理解
C++ 函数重载如何与虚函数结合使用?
2024-04-13

python如何使用sample()函数从指定序列中随机获取指定长度的片段

这篇文章给大家分享的是有关python如何使用sample()函数从指定序列中随机获取指定长度的片段的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。sample(sequence,k)从指定序列中随机获取指定长度的
2023-06-08

C++ 函数重载中的 best match 是如何决定的?

最佳匹配的确定顺序是:精度匹配标准转换用户定义转换默认参数数量最少C++ 函数重载中的最佳匹配是如何决定的?在 C++ 中,函数重载允许您创建具有相同名称但参数不同的多个函数版本。编译器在使用适当的重载版本时,遵循一套规则来确定最佳匹配。
C++ 函数重载中的 best match 是如何决定的?
2024-04-26

如何在 C++ 中有效使用函数重载和重写

函数重载与重写指南:函数重载: 创建多个具有相同名称但不同参数的函数,减少代码冗余。函数重写: 子类中声明同名函数,修改继承函数的行为,实现多态。实战案例:函数重载:处理不同数据类型。函数重写:实现继承多态。最佳实践:仔细考虑重载函数签名。
如何在 C++ 中有效使用函数重载和重写
2024-04-20

C++ 函数重载中如何使用宏来简化代码?

宏简化 c++++ 函数重载:创建宏,将通用代码提取到单个定义中。在每个重载函数中使用宏替换通用的代码部分。实际应用包括创建打印输入数据类型信息的函数,分别处理 int、double 和 string 数据类型。利用宏简化 C++ 函数重载
C++ 函数重载中如何使用宏来简化代码?
2024-04-13

如何使用 PHP 函数获取给定月份的最后一天

PHP 已经具有内置函数来获取给定月份的最后一天的日期。在这种情况下,DateTime 和 strtotime 函数是最有用的,尽管众所周知并已写入文档了 strtotime 对于 2038 年之后的日期不能正确地工作或给出正确的结果。但是
如何使用 PHP 函数获取给定月份的最后一天
2024-02-27

如何使用C#中的Math.Pow函数计算指定数的幂次方

在C#中,有一个Math类库,其中包含许多数学函数。其中包括计算幂次方的函数Math.Pow,它可以帮助我们计算指定数的幂。Math.Pow函数的用法非常简单,只需要指定底数和指数就可以了。其语法如下:Math.Pow(base, expo
如何使用C#中的Math.Pow函数计算指定数的幂次方
2023-11-18

如何使用C#中的Enum.GetNames函数获取枚举类型中所有定义的名称

在C#中,枚举类型是一种非常有用的数据类型,它允许我们定义一些常量来表示某些状态或选项。一旦定义了枚举类型,就可以使用Enum.GetNames函数来获取所有定义的名称。本文将详细介绍如何使用该函数,并提供具体的代码示例。什么是Enum.G
如何使用C#中的Enum.GetNames函数获取枚举类型中所有定义的名称
2023-11-18

如何使用 C++ 函数模板实现函数指针的泛型化?

c++++ 函数模板允许泛化函数指针,支持不同类型参数的函数指针。具体步骤如下:声明带有函数指针的函数模板,其中 t 为模板类型参数。将要泛化的函数指针作为参数传递给模板函数。模板函数返回泛型函数指针。使用 C++ 函数模板实现函数指针的泛
如何使用 C++ 函数模板实现函数指针的泛型化?
2024-04-15

如何使用golang中的os.Stat函数获取文件的信息

如何使用golang中的os.Stat函数获取文件的信息,需要具体代码示例作为一种现代化的编程语言,Golang(也称为Go)为开发者提供了丰富的标准库,其中包含了丰富的文件和目录操作功能,满足了大多数编程需求。其中,os包是Golang中
如何使用golang中的os.Stat函数获取文件的信息
2023-11-18

如何使用 MySQL FIND_IN_SET() 函数从表中获取特定记录作为结果集

要使用 MySQL 的 FIND_IN_SET() 函数从表中获取特定记录作为结果集,可以按照以下步骤进行操作:1. 确保已经连接到 MySQL 数据库。2. 编写 SQL 查询语句,使用 FIND_IN_SET() 函数来筛选特定记录。语
2023-10-20

如何使用 MySQL FIND_IN_SET() 函数从表中获取特定记录作为结果集?

我们可以通过提供特定字符串和列名称作为 FIND_IN_SET() 函数的参数来获取记录作为结果集。我们还需要将 WHERE 子句与 FIND_IN_SET() 函数一起使用。为了理解它,我们使用表“student_info”中的数据,如下
2023-10-22

如何使用C#中的File.ReadAllText函数读取文本文件内容

如何使用C#中的File.ReadAllText函数读取文本文件内容在C#编程中,我们经常需要读取文本文件的内容。File.ReadAllText是一个非常方便的函数,可以帮助我们快速读取文本文件的全部内容。本文将介绍如何使用File.Re
如何使用C#中的File.ReadAllText函数读取文本文件内容
2023-11-18

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录