将TForm作为参数传递给函数

如何解决将TForm作为参数传递给函数

我有一个带有几种表格的应用程序。它们中的两个非常相似,它们具有共同的VCL对象(标签,图像等)形式的功能,我将其命名为相同。 我想在特定的类中有一个函数,该函数可以接受这两个Form之一作为参数,以便修改它们共同具有的参数。我遇到的解决方案似乎不起作用。

由于我的应用程序很大且很复杂,所以我用一个小例子重现了该问题。 首先,下面是我的MainForm的示例:

enter image description here

还有一个子表单的示例(它们都以类似的方式排列)

enter image description here

我有一个additional类,该类用于填写子表单上的“编辑”。此类的代码如下:

#pragma hdrstop

#include "master_class.h"
#include "sub_Form2.h"
#include "sub_Form3.h"
#include "sub_Form4.h"
//---------------------------------------------------------------------------
#pragma package(smart_init)

Master::Master(void)
{

}

Master::~Master(void)
{

 }

 void Master::WriteToForm(TForm* Form)
{
TForm2* curForm = static_cast<TForm2*>(Form);

TForm3* self = dynamic_cast<TForm3*>(Form);
TForm2* self2 = dynamic_cast<TForm2*>(Form);

if (self != NULL && self2 == NULL) {
    TForm3* curForm = static_cast<TForm3*>(Form);
}

else if (self == NULL && self2 != NULL) {
    TForm2* curForm = static_cast<TForm2*>(Form);
}

curForm -> Edit1 -> Text = "blablabla_1";
curForm -> Edit2 -> Text = "blablabla_2";
}

在MainForm中,“填写Form2”按钮的代码如下:

Master1 -> WriteToForm(Form2);

其中Master1只是Master类的对象。 这对于Form2来说效果很好:

enter image description here

但是对于使用 Master1 -> WriteToForm(Form3)填充的Form3,这是我得到的,与实际应用程序中的pb相同。

enter image description here

所以应该去编辑的东西放错了位置。我认为主要的pb来自以下事实:我没有按相同的顺序创建每个标签,进行编辑等。我这样做是为了模仿我的真实应用。为了验证这一点,我创建了第三个子窗体,这次,以与我的第一个子窗体相同的顺序创建了VCL对象,并且可以正常工作:

enter image description here

所以我怀疑这是来自最初的演员

TForm2* curForm = static_cast<TForm2*>(Form);

当我将Form3作为参数传递时,Form3在某种程度上被转换为Form2的“形状”,该形状未按相同的顺序定义。也许可以通过直接修改DFM文件来纠正此问题,但这对我的主应用程序来说不是一个现实的方法。

我执行此初始转换,否则我收到编译错误,说第一行不知道curForm

curForm -> Edit1 -> Text = "blablabla_1";

因此,是否有更好的方法将Form作为参数传递给WriteToForm函数?

解决方法

仅仅因为两种类型是相似并不意味着它们是相关。您的代码不起作用,因为您的两个Form类之间没有任何关联。您不能随便丢一个。

要解决此问题,您有几种选择:

  1. 分别为两个Form类编码,例如:
void Master::WriteToForm(TForm* Form)
{
    TForm2* curForm2 = dynamic_cast<TForm2*>(Form);
    TForm3* curForm3 = dynamic_cast<TForm3*>(Form);

    if (curForm2)
    {
        curForm2->Edit1->Text = _D("blablabla_1");
        curForm2->Edit2->Text = _D("blablabla_2");
    }
    else if (curForm3)
    {
        curForm3->Edit1->Text = _D("blablabla_1");
        curForm3->Edit2->Text = _D("blablabla_2");
    }
}

或者:

void WriteToForm(TForm2* Form);
void WriteToForm(TForm3* Form);

...

void Master::WriteToForm(TForm2* Form)
{
    Form->Edit1->Text = _D("blablabla_1");
    Form->Edit2->Text = _D("blablabla_2");
}

void Master::WriteToForm(TForm3* Form)
{
    Form->Edit1->Text = _D("blablabla_1");
    Form->Edit2->Text = _D("blablabla_2");
}
  1. 使您的功能使用模板(不过,请注意:Why can templates only be implemented in the header file?):
template<typename T>
void WriteToForm(T* Form);

...

void Master::WriteToForm<T>(T* Form)
{
    Form->Edit1->Text = _D("blablabla_1");
    Form->Edit2->Text = _D("blablabla_2");
}
  1. 使这两个Form类从公共基类或接口派生,例如:
class TBaseForm : public TForm
{
public:
    inline __fastcall TBaseForm(TComponent *Owner) : TForm(Owner) {}
    virtual void SetEdit1(const String &Text) = 0;
    virtual void SetEdit2(const String &Text) = 0;
};

...

class TForm2 : public TBaseForm
{
    ...
public:
    __fastcall TForm2(TComponent *Owner);
    ...
    void SetEdit1(const String &NewText);
    void SetEdit2(const String &NewText);
};

__fastcall TForm2::TForm2(TComponent *Owner)
    : TBaseForm(Owner)
{
    ...
}

void TForm2::SetEdit1(const String &NewText)
{
    Edit1->Text = NewText;
}

void TForm2::SetEdit2(const String &NewText)
{
    Edit2->Text = NewText;
}

...

repeat for TForm3...

...

void Master::WriteToForm(TBaseForm* Form)
{
    Form->SetEdit1(_D("blablabla_1"));
    Form->SetEdit2(_D("blablabla_2"));
}

或者:

__interface INTERFACE_UUID("{E900785E-0151-480F-A33A-1F1452A431D2}")
IMyIntf : public IInterface
{
public:
    virtual void SetEdit1(const String &Text) = 0;
    virtual void SetEdit2(const String &Text) = 0;
};

...

class TForm2 : public TForm,public IMyIntf
{
    ...
public:
    __fastcall TForm2(TComponent *Owner);
    ...
    void SetEdit1(const String &NewText);
    void SetEdit2(const String &NewText);
};

__fastcall TForm2::TForm2(TComponent *Owner)
    : TForm(Owner)
{
    ...
}

void TForm2::SetEdit1(const String &NewText)
{
    Edit1->Text = NewText;
}

void TForm2::SetEdit2(const String &NewText)
{
    Edit2->Text = NewText;
}

...

repeat for TForm3...

...

void Master::WriteToForm(IMyIntf* Intf)
{
    Intf->SetEdit1(_D("blablabla_1"));
    Intf->SetEdit2(_D("blablabla_2"));
}
  1. 使用RTTI访问字段,例如:
#include <System.Rtti.hpp>

void Master::WriteToForm(TForm* Form)
{
    TRttiContext Ctx;
    TRttiType *FormType = Ctx.GetType(Form->ClassType());

    TRttiField *Field = FormType->GetField(_D("Edit1"));
    if (Field)
    {
        TValue value = Field->GetValue(Form);
        if( (!value.Empty) && (value.IsObject()) )
        {
            TObject *Obj = value.AsObject();

            // Either: 

            static_cast<TEdit*>(Obj)->Text = _D("blablabla_1");

            // Or:

            TRttiProperty *Prop = Ctx.GetType(Obj->ClassType())->GetProperty(_D("Text"));
            if (Prop) Prop->SetValue(Obj,String(_D("blablabla_1")));
        }
    }

    Field = FormType->GetField(_D("Edit2"));
    if (Field)
    {
        TValue value = Field->GetValue(Form);
        if( (!value.Empty) && (value.IsObject()) )
        {
            TObject *Obj = value.AsObject();

            // Either: 

            static_cast<TEdit*>(Obj)->Text = _D("blablabla_2");

            // Or:

            TRttiProperty *Prop = Ctx.GetType(Obj->ClassType())->GetProperty(_D("Text"));
            if (Prop) Prop->SetValue(Obj,String(_D("blablabla_2")));
        }
    }
}

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 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