铸造成员函数指针

如何解决铸造成员函数指针

| 我需要使用一个成员函数指针,该指针接受其他代码中使用的基类的参数。好吧,我只是想做下面的例子。这段代码可以正常工作,但是我想知道这样的转换是否总是安全的吗?我不能在这里做
dynamic
static
#include <cstdio>                                                   

class C
{                                                           
public:                                                             
        C () : c(\'c\') {}                                            
        virtual ~C() {}                                             

        const char c;                                               
};                                                                  

class D : public C
{                                                
public:                                                             
        D () : d(\'d\') {}                                            
        virtual ~D() {}                                             

        const char d;                                               
};                                                                  

class A 
{                                                           
public:                                                             
        A () {}                                                     
        virtual ~A() {}                                             

        void f( C& c ) { printf(\"%c\\n\",c.c); }                      
        void g( D& d ) { printf(\"%c %c\\n\",d.c,d.d); }               
};                                                                  

int main (int argc,char const* argv[])                             
{                                                                   
        void (A::*pf)( C& c ) = &A::f;                              
        void (A::*pg)( D& d ) = reinterpret_cast<void (A::*)(D&)>(&A::f);

        A a;                                                        
        C c;                                                        
        D d;                                                        

        (a.*pf)( c );                                               
        (a.*pg)( d );                                               

        return 0;                                                   
}                                                              
    

解决方法

        您要尝试执行的操作无法在C ++中合法完成。 C ++不支持函数参数类型的任何协方差或反方差,无论这是成员函数还是自由函数。 在您的情况下,实现它的正确方法是为参数类型转换目的引入一个中间函数
class A 
{                                                           
public:          
  ...                                                   
  void f( C& c ) { printf(\"%c\\n\",c.c); }                      
  void f_with_D( D& d ) { f(d); }
  ...
};          
并使您的指针指向该中间函数而无需任何强制转换
void (A::*pg)( D& d ) = &A::f_with_D;
现在
A a;
D d;                                                        
(a.*pg)( d );
最终将对象
d
C
子对象作为参数调用
a.f
。 编辑:是的,它将与函数重载一起工作(如果我正确理解您的问题)。您只需要记住,对于函数重载,为了将内部调用定向到函数的正确版本,您将必须使用显式强制转换
class A 
{                                                           
public:          
  ...                                                   
  void f( C& c ) { printf(\"%c\\n\",c.c); }                      
  void f( D& d ) { f(static_cast<C&>(d)); }
  ...
};          
如果不进行强制转换,最终将以
A::f(D&)
递归调用自身。     ,        不,您的示例无法正常工作。 首先,您只能使用ѭ11来在相关的类类型之间进行转换,而不能使用其他类型。 其次,即使您将ѭ11替换为cast13 cast或C样式的强制转换(我认为是您的意思),我也会得到以下输出:   C   C 并不是您真正想要的。 为什么这样做甚至不会造成可怕的崩溃,是因为在成员函数指针之间来回投射是“安全的”,不会丢失任何信息。 之所以仍然打印某些内容,是因为编译器没有看到类型错误,但是程序集并不关心类型,只关心地址,因此仍会调用call14ѭ,因为这是您保存的指针,无论方式。 有趣的是,即使您取消类的关联(
D
也不继承自
C
),这仍然有效,同样是因为汇编并不关心类型。通过以下方式更改A中的功能:
void f( C& c ) { printf(\"f(C& c): %c\\n\",c.c); }
void g( D& d ) { printf(\"g(D& d): %c\\n\",d.d); }
导致以下输出:   f(C&c):c   f(C&c):d \“如何运作?
D
甚至没有a19ѭ会员!\”。好吧,再次因为地址。两个变量与
this
指针的偏移量相同,即
+0
。现在,让另一个成员进入
C
(简化的类):
struct C{
        C () : c(\'c\') {}
        int i; // mean
        const char c;
};
然后再试一次,输出:   f(C&c):c   f(C&c):╠ 是的,我们去了。 24ѭ现在在偏移
+4
+0
+
sizeof int
)处,ѭ28from从那里读取。在
D
中,没有这样的偏移,
printf
从未初始化的存储器中读取。另一方面,访问未初始化的内存是未定义的行为。 因此,最后得出结论:不,这不安全。 :)     ,        编译器应拒绝您编写的带有dynamic_cast的代码。 (我认为这是一个错字,考虑到介绍文字,您的意思是reinterpret_cast)。 使用reinterpret_cast,您就不会处于定义明确的情况之一(多数情况下涉及转换为另一种类型,然后再转换为原始类型)。因此,对于转换结果,我们处于未指定的领域,对于调用转换结果时的行为,我们处于未定义的领域。     ,        您需要使用
reinterpret_cast
使其起作用。在这种情况下,它应该是安全的(请参见备注),但是如果使用多重继承,则可能会失败,因为在将ѭ15传递为ѭ7时需要调整指针。编译器需要知道这是必须发生的,在这种情况下是不可能的(用
d
调用
pg
会跳过此步骤,成员函数将获得
D
对象的未修改地址)。 备注:我说的很安全-好吧,实际上这是未定义的行为,这是因为将一个类型重新解释为不相关的类型并使用该类型,但是它仍然可以在大多数编译器上运行。拜托,请不要在生产代码中这样做。     

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams[&#39;font.sans-serif&#39;] = [&#39;SimHei&#39;] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -&gt; systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping(&quot;/hires&quot;) public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate&lt;String
使用vite构建项目报错 C:\Users\ychen\work&gt;npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-
参考1 参考2 解决方案 # 点击安装源 协议选择 http:// 路径填写 mirrors.aliyun.com/centos/8.3.2011/BaseOS/x86_64/os URL类型 软件库URL 其他路径 # 版本 7 mirrors.aliyun.com/centos/7/os/x86
报错1 [root@slave1 data_mocker]# kafka-console-consumer.sh --bootstrap-server slave1:9092 --topic topic_db [2023-12-19 18:31:12,770] WARN [Consumer clie
错误1 # 重写数据 hive (edu)&gt; insert overwrite table dwd_trade_cart_add_inc &gt; select data.id, &gt; data.user_id, &gt; data.course_id, &gt; date_format(
错误1 hive (edu)&gt; insert into huanhuan values(1,&#39;haoge&#39;); Query ID = root_20240110071417_fe1517ad-3607-41f4-bdcf-d00b98ac443e Total jobs = 1
报错1:执行到如下就不执行了,没有显示Successfully registered new MBean. [root@slave1 bin]# /usr/local/software/flume-1.9.0/bin/flume-ng agent -n a1 -c /usr/local/softwa
虚拟及没有启动任何服务器查看jps会显示jps,如果没有显示任何东西 [root@slave2 ~]# jps 9647 Jps 解决方案 # 进入/tmp查看 [root@slave1 dfs]# cd /tmp [root@slave1 tmp]# ll 总用量 48 drwxr-xr-x. 2
报错1 hive&gt; show databases; OK Failed with exception java.io.IOException:java.lang.RuntimeException: Error in configuring object Time taken: 0.474 se
报错1 [root@localhost ~]# vim -bash: vim: 未找到命令 安装vim yum -y install vim* # 查看是否安装成功 [root@hadoop01 hadoop]# rpm -qa |grep vim vim-X11-7.4.629-8.el7_9.x
修改hadoop配置 vi /usr/local/software/hadoop-2.9.2/etc/hadoop/yarn-site.xml # 添加如下 &lt;configuration&gt; &lt;property&gt; &lt;name&gt;yarn.nodemanager.res