gtkmm 小部件 - 使用智能指针还是指针?

如何解决gtkmm 小部件 - 使用智能指针还是指针?

我正在尝试学习如何使用 gtkmm 并基本掌握了 C++(我喜欢挑战!)。我一直在学习教程(以及其他阅读材料)。我正在尝试使用使用glade的方法来设计UI,然后编写代码来完成工作。

所以我构建了一个非常简单的用户界面(目前是窗口和按钮!)。我正在使用 GTK::Builder 从文件加载 UI。我将代码分为类和主要调用者。

这里是main.cpp

#include "hellowindow.h"
#include <gtkmm/application.h>

int main(int argc,char *argv[]) {
    auto app = Gtk::Application::create(argc,argv,"org.gtkmm.example"); //creates a Gtk::Application object,stored in a Glib::RefPtr smartpointer,create() method for this object initializes gtkmm.
    HelloWindow hw; // Create a HelloWindow object
    return app->run(hw,argc,argv); // shows the HelloWindow object and enter the gtkmm main processing loop,will then return with an appropriate success or error code
}

这是 HelloWindow 类的标题

#ifndef HELLOWINDOW_H
#define HELLOWINDOW_H

#include <gtkmm/application.h>
#include <gtkmm/applicationwindow.h>
#include <gtkmm/button.h>
#include <gtkmm/box.h>
#include <gtkmm/builder.h>
#include <glibmm/fileutils.h>

/* derive the class from Gtk::ApplicationWindow base class */
class HelloWindow : public Gtk::ApplicationWindow {

public:
  /* Conctructor */
  HelloWindow();

  /* Destructor */
  ~HelloWindow() override;

protected:
  /* Signal handlers: */
  void on_button_clicked();

  /* Member widgets: */
  Gtk::Box *cont;                             // Container
  Gtk::Button *pButton;                       // Pointer to a Button
  Glib::RefPtr<Gtk::Button> display_btn;      // Smart pointer to a Button
  Glib::RefPtr<Gtk::Builder> builder;         // Builder
};

#endif // HELLOWINDOW_H

这是课程代码:

#include "hellowindow.h"
#include <iostream>

HelloWindow::HelloWindow() : builder(Gtk::Builder::create()){
  try {
    /* load window from glade file */
    builder->add_from_file("glade/simple.glade");
  }
  catch(const Glib::FileError& ex) {
    /* catch file errors */
    std::cerr << "FileError: " << ex.what() << std::endl;
    return;
  }
  /* ui builder  created successfully from file */
  /* add a container to the builder */
  builder->get_widget<Gtk::Box>("cont",cont);

  builder->get_widget<Gtk::Button>("display_button",pButton);

  pButton->signal_clicked().connect(
    sigc::mem_fun(*this,&HelloWindow::on_button_clicked)
  );

  /* add the container to the application window */
  add(*cont);

  /* set some parameters for the window */
  set_title("Simple Gtk::Builder Demo"); // set the window title
  set_default_size(500,500); // set the window size
  show_all(); // show the window and all of the enclosed widgets
}

HelloWindow::~HelloWindow(){
}

void HelloWindow::on_button_clicked(){
  std::cout << "Hello World" << std::endl;
}

这一切正常,我想我明白发生了什么。但是,我看到了一种在运行时添加小部件的不同方法 (https://sodocumentation.net/gtk3/topic/5579/using-glade-with-builder-api)。不同之处在于按钮对象的声明方式。在上面的代码中,它被声明为指向按钮对象的指针:

builder->get_widget<Gtk::Button>("display_button",pButton);

然而,上面的网站使用了一种指向按钮对象的智能指针的方法:

display_btn = Glib::RefPtr<Gtk::Button>::cast_dynamic(builder->get_object("display_button"));

第二种方法似乎不太清楚,特别是 cast_dynamic 方面,有人能解释一下这两种方法之间的区别吗?

我希望我已经包含了足够的信息。

谢谢

马丁

解决方法

首先,要了解区别,而且由于您是新手 C++,我建议阅读以下主题:

第一种方法:get_widget

通过这种方法,您将获得一个原始指针(而不是智能指针) 指针)指向从 ui 文件定义的 widget。示例:

Gtk::Grid* pGrid = nullptr;
refXml->get_widget("mygrid",pGrid);

此后,pGrid 指向一个 Gtk::Grid 小部件。请注意,没有 需要强制转换,您会立即获得 Gtk::Grid。这是因为 Gtk::Builder::get_widget 方法是一个模板方法:

// T_Widget is like a placeholder for some widget type,// like Gtk::Grid for example.
template <class T_Widget >
void Gtk::Builder::get_widget(const Glib::ustring& name,T_Widget*& widget 
                              )

通常在 C++ 中,例如 原始指针可能很危险,因为如果它们指向一个对象 在堆上分配(通常使用new),必须记住使用 delete 完成后,否则会发生内存泄漏。在这 在这种情况下,pGrid 确实是指向分配在 堆,但是 documentation states:

请注意,您有责任删除顶级小部件(windows 和对话框)由 Builder 对象实例化。其他小部件是 实例化为托管,因此如果您将它们自动删除 将它们添加到容器小部件中。

所以大部分时间(即如果不是顶级小部件),您没有 调用 delete,但有时你会这样做。这是因为 Gtkmm 有设施 自动 delete 销毁的对象。参见Gtk::manage 有关这方面的更多信息。

何时使用或不使用 delete 可能会随着代码的增长而变得困难,尤其是因为 您不必总是这样做(很容易忘记)。

第二种方法:get_object

通过这种方法,您将获得一个智能指针(Glib::RefPtr) 需要一个 object 并转换为正确的类型,因此 cast_dynamic

// get_object returns a Glib::RefPtr<Glib::Object>,which is not a Glib::RefPtr<Gtk::Button>
// so a cast is performed. This works because Gtk::Button is a child class of
// Glib::Object.
display_btn = Glib::RefPtr<Gtk::Button>::cast_dynamic(builder->get_object("display_button"));

这样做的好处是一旦进行了演员表, 您不必管理对象的内存。它由 智能指针(即智能指针会自动调用 delete 为你。即使对于“顶级”小部件也是如此。

注意:这里使用的演员表 cast_dynamic 是一个 Gtkmm 特定的演员表,它包含一个 dynamic_cast

我的看法

我个人会采用第二种方法,因为你会自动 内存管理(即使是顶级小部件),因此没有内存泄漏。 但是,正如您已经注意到的那样,代码变得更难阅读了。

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