Gobject tutorial 三

Derivable type and abstract type

Derivable type

在GLib中,类型可以分为两种。一种是可以被继承的(derivable),一种是不能被继承的(final)。二者的主要区别是,final 类型对象的类结构中,除了其父类外,再无其他成员。而derivable类型对象的类结构中会存在其他成员, 且derivable类型的类对其衍生类是可见的。对于上篇我们的例子TDouble,就是一个final类。我们在注册时使用的是G_DEFINE_TYPE,实际上还可以使用G_DECLARE_FINAL_TYPE。

Abstract type

abstract type 属于derivable type,只是abstract type 不会生成实例,但其衍生对象可以使用其函数和信号。

对于abstract type,我们将会创建TNumber类型来进行说明。上篇我们说TDouble 代表了浮点数,那么,更抽象一层,TNumber则表示全体数。TNumber是TDouble 的父类,由于TNumber属于abstract type,因此不能按照我们实例化TDouble那样进行实例化。毕竟不存在能够对数字进行统一表示的方式。int 类型的5.0,与double类型的5.0,虽然绝对值是一样的,但是int类型的5.0相对于double类型的5.0,其实是丧失精度的,因此两者并不相同。但任何包含TDouble的对象在实例化时,也意味着TNumber也被实例化了。例如,当TDouble中value为5.0,我们可以说5.0就是TDouble的实例,同时,5.0首先是个数字,因此,5.0也是TNumber的实例化。

TNumber class

G_DECLARE_DERIVABLE_TYPE (TNumber, t_number, T, NUMBER, GObject)struct _TNumberClass {GObjectClass parent_class;TNumber* (*add) (TNumber *self, TNumber *other);TNumber* (*sub) (TNumber *self, TNumber *other);TNumber* (*mul) (TNumber *self, TNumber *other);TNumber* (*div) (TNumber *self, TNumber *other);TNumber* (*uminus) (TNumber *self);char * (*to_s) (TNumber *self);/* signal */void (*div_by_zero) (TNumber *self);
};

 在glib 2.78.2中,宏G_DECLARE_DERIVABLE_TYPE的定义如下:

#define G_DECLARE_DERIVABLE_TYPE(ModuleObjName, module_obj_name, MODULE, OBJ_NAME, ParentName) \GType module_obj_name##_get_type (void);                                                               \G_GNUC_BEGIN_IGNORE_DEPRECATIONS                                                                       \typedef struct _##ModuleObjName ModuleObjName;                                                         \typedef struct _##ModuleObjName##Class ModuleObjName##Class;                                           \struct _##ModuleObjName { ParentName parent_instance; };                                               \\_GLIB_DEFINE_AUTOPTR_CHAINUP (ModuleObjName, ParentName)                                               \G_DEFINE_AUTOPTR_CLEANUP_FUNC (ModuleObjName##Class, g_type_class_unref)                               \\G_GNUC_UNUSED static inline ModuleObjName * MODULE##_##OBJ_NAME (gpointer ptr) {                       \return G_TYPE_CHECK_INSTANCE_CAST (ptr, module_obj_name##_get_type (), ModuleObjName); }             \G_GNUC_UNUSED static inline ModuleObjName##Class * MODULE##_##OBJ_NAME##_CLASS (gpointer ptr) {        \return G_TYPE_CHECK_CLASS_CAST (ptr, module_obj_name##_get_type (), ModuleObjName##Class); }         \G_GNUC_UNUSED static inline gboolean MODULE##_IS_##OBJ_NAME (gpointer ptr) {                           \return G_TYPE_CHECK_INSTANCE_TYPE (ptr, module_obj_name##_get_type ()); }                            \G_GNUC_UNUSED static inline gboolean MODULE##_IS_##OBJ_NAME##_CLASS (gpointer ptr) {                   \return G_TYPE_CHECK_CLASS_TYPE (ptr, module_obj_name##_get_type ()); }                               \G_GNUC_UNUSED static inline ModuleObjName##Class * MODULE##_##OBJ_NAME##_GET_CLASS (gpointer ptr) {    \return G_TYPE_INSTANCE_GET_CLASS (ptr, module_obj_name##_get_type (), ModuleObjName##Class); }       \G_GNUC_END_IGNORE_DEPRECATIONS

 对于TNumber对象来说,G_DECLARE_DERIVABLE_TYPE宏的作用如下:

  • 声明函数t_number_get_type()。
  • 定义TNumber数据结构,此数据结构的成员只有其父实例。
  • 声明TNumberClass数据结构,具体定义需要用户实现。
  • 定义宏T_NUMBER(cast to instance)、T_NUMBER_CLASS(cast to class)、T_IS_NUMBER(instance check)、T_IS_NUMBER_CLASS(class check)、T_NUMBER_GET_CLASS.
  • g_autoptr()。

 struct _TNumber中的函数指针称为类方法或者虚拟函数,他们会被TNumber的衍生对象重写。

对照上篇所说的G_DEFINE_TYPE宏的工作,对于abstract type,有个同样功能的宏G_DEFINE_ABSTRACT。

TInt object

介绍完Derivable type和Abstract type,现在我们来介绍其应用。我们使用新类型TInt来说明。

/*tint.h*/
#pragma once#include <glib-object.h>#define T_TYPE_INT  (t_int_get_type ())
G_DECLARE_FINAL_TYPE (TInt, t_int, T, INT, TNumber)/* create a new TInt instance */
TInt *
t_int_new_with_value (int value);TInt *
t_int_new (void);
/*tint.c*/
#include "tnumber.h"
#include "tint.h"
#include "tdouble.h"#define PROP_INT 1
static GParamSpec *int_property = NULL;struct _TInt {TNumber parent;int value;
};G_DEFINE_TYPE (TInt, t_int, T_TYPE_NUMBER)static void
t_int_set_property (GObject *object, guint property_id, const GValue *value, GParamSpec *pspec) {TInt *self = T_INT (object);if (property_id == PROP_INT)self->value = g_value_get_int (value);elseG_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}static void
t_int_get_property (GObject *object, guint property_id, GValue *value, GParamSpec *pspec) {TInt *self = T_INT (object);if (property_id == PROP_INT)g_value_set_int (value, self->value);elseG_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}static void
t_int_init (TInt *self) {
}/* arithmetic operator */
/* These operators create a new instance and return a pointer to it. */
#define t_int_binary_op(op) \int i; \double d; \if (T_IS_INT (other)) { \g_object_get (T_INT (other), "value", &i, NULL); \return  T_NUMBER (t_int_new_with_value (T_INT(self)->value op i)); \} else { \g_object_get (T_DOUBLE (other), "value", &d, NULL); \return  T_NUMBER (t_int_new_with_value (T_INT(self)->value op (int) d)); \}static TNumber *
t_int_add (TNumber *self, TNumber *other) {g_return_val_if_fail (T_IS_INT (self), NULL);t_int_binary_op (+)
}static TNumber *
t_int_sub (TNumber *self, TNumber *other) {g_return_val_if_fail (T_IS_INT (self), NULL);t_int_binary_op (-)
}static TNumber *
t_int_mul (TNumber *self, TNumber *other) {g_return_val_if_fail (T_IS_INT (self), NULL);t_int_binary_op (*)
}static TNumber *
t_int_div (TNumber *self, TNumber *other) {g_return_val_if_fail (T_IS_INT (self), NULL);int i;double d;if (T_IS_INT (other)) {g_object_get (T_INT (other), "value", &i, NULL);if (i == 0) {g_signal_emit_by_name (self, "div-by-zero");return NULL;} elsereturn  T_NUMBER (t_int_new_with_value (T_INT(self)->value / i));} else {g_object_get (T_DOUBLE (other), "value", &d, NULL);if (d == 0) {g_signal_emit_by_name (self, "div-by-zero");return NULL;} elsereturn  T_NUMBER (t_int_new_with_value (T_INT(self)->value / (int)  d));}
}static TNumber *
t_int_uminus (TNumber *self) {g_return_val_if_fail (T_IS_INT (self), NULL);return T_NUMBER (t_int_new_with_value (- T_INT(self)->value));
}static char *
t_int_to_s (TNumber *self) {g_return_val_if_fail (T_IS_INT (self), NULL);int i;g_object_get (T_INT (self), "value", &i, NULL); return g_strdup_printf ("%d", i);
}static void
t_int_class_init (TIntClass *class) {TNumberClass *tnumber_class = T_NUMBER_CLASS (class);GObjectClass *gobject_class = G_OBJECT_CLASS (class);/* override virtual functions */tnumber_class->add = t_int_add;tnumber_class->sub = t_int_sub;tnumber_class->mul = t_int_mul;tnumber_class->div = t_int_div;tnumber_class->uminus = t_int_uminus;tnumber_class->to_s = t_int_to_s;gobject_class->set_property = t_int_set_property;gobject_class->get_property = t_int_get_property;int_property = g_param_spec_int ("value", "val", "Integer value", G_MININT, G_MAXINT, 0, G_PARAM_READWRITE);g_object_class_install_property (gobject_class, PROP_INT, int_property);
}TInt *
t_int_new_with_value (int value) {TInt *i;i = g_object_new (T_TYPE_INT, "value", value, NULL);return i;
}TInt *
t_int_new (void) {TInt *i;i = g_object_new (T_TYPE_INT, NULL);return i;
}

TDouble object

我们对TDouble进行重构,以增加其功能,使TDouble类型的数据能够使用TNumber的函数进行基础运算。

/*tdouble.h*/
#pragma once#include <glib-object.h>#define T_TYPE_DOUBLE  (t_double_get_type ())
G_DECLARE_FINAL_TYPE (TDouble, t_double, T, DOUBLE, TNumber)/* create a new TDouble instance */
TDouble *
t_double_new_with_value (double value);TDouble *
t_double_new (void);
/*tdouble.c*/
#include "tnumber.h"
#include "tdouble.h"
#include "tint.h"#define PROP_DOUBLE 1
static GParamSpec *double_property = NULL;struct _TDouble {TNumber parent;double value;
};G_DEFINE_TYPE (TDouble, t_double, T_TYPE_NUMBER)static void
t_double_set_property (GObject *object, guint property_id, const GValue *value, GParamSpec *pspec) {TDouble *self = T_DOUBLE (object);if (property_id == PROP_DOUBLE) {self->value = g_value_get_double (value);} elseG_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}static void
t_double_get_property (GObject *object, guint property_id, GValue *value, GParamSpec *pspec) {TDouble *self = T_DOUBLE (object);if (property_id == PROP_DOUBLE)g_value_set_double (value, self->value);elseG_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
}static void
t_double_init (TDouble *self) {
}/* arithmetic operator */
/* These operators create a new instance and return a pointer to it. */
#define t_double_binary_op(op) \int i; \double d; \if (T_IS_INT (other)) { \g_object_get (T_INT (other), "value", &i, NULL); \return  T_NUMBER (t_double_new_with_value (T_DOUBLE(self)->value op (double) i)); \} else { \g_object_get (T_DOUBLE (other), "value", &d, NULL); \return  T_NUMBER (t_double_new_with_value (T_DOUBLE(self)->value op d)); \}static TNumber *
t_double_add (TNumber *self, TNumber *other) {g_return_val_if_fail (T_IS_DOUBLE (self), NULL);t_double_binary_op (+)
}static TNumber *
t_double_sub (TNumber *self, TNumber *other) {g_return_val_if_fail (T_IS_DOUBLE (self), NULL);t_double_binary_op (-)
}static TNumber *
t_double_mul (TNumber *self, TNumber *other) {g_return_val_if_fail (T_IS_DOUBLE (self), NULL);t_double_binary_op (*)
}static TNumber *
t_double_div (TNumber *self, TNumber *other) {g_return_val_if_fail (T_IS_DOUBLE (self), NULL);int i;double d;if (T_IS_INT (other)) {g_object_get (T_INT (other), "value", &i, NULL);if (i == 0) {g_signal_emit_by_name (self, "div-by-zero");return NULL;} elsereturn  T_NUMBER (t_double_new_with_value (T_DOUBLE(self)->value / (double) i));} else {g_object_get (T_DOUBLE (other), "value", &d, NULL);if (d == 0) {g_signal_emit_by_name (self, "div-by-zero");return NULL;} elsereturn  T_NUMBER (t_double_new_with_value (T_DOUBLE(self)->value / d));}
}static TNumber *
t_double_uminus (TNumber *self) {g_return_val_if_fail (T_IS_DOUBLE (self), NULL);return T_NUMBER (t_double_new_with_value (- T_DOUBLE(self)->value));
}static char *
t_double_to_s (TNumber *self) {g_return_val_if_fail (T_IS_DOUBLE (self), NULL);double d;g_object_get (T_DOUBLE (self), "value", &d, NULL);return g_strdup_printf ("%lf", d);
}static void
t_double_class_init (TDoubleClass *class) {TNumberClass *tnumber_class = T_NUMBER_CLASS (class);GObjectClass *gobject_class = G_OBJECT_CLASS (class);/* override virtual functions */tnumber_class->add = t_double_add;tnumber_class->sub = t_double_sub;tnumber_class->mul = t_double_mul;tnumber_class->div = t_double_div;tnumber_class->uminus = t_double_uminus;tnumber_class->to_s = t_double_to_s;gobject_class->set_property = t_double_set_property;gobject_class->get_property = t_double_get_property;double_property = g_param_spec_double ("value", "val", "Double value", -G_MAXDOUBLE, G_MAXDOUBLE, 0, G_PARAM_READWRITE);g_object_class_install_property (gobject_class, PROP_DOUBLE, double_property);
}TDouble *
t_double_new_with_value (double value) {TDouble *d;d = g_object_new (T_TYPE_DOUBLE, "value", value, NULL);return d;
}TDouble *
t_double_new (void) {TDouble *d;d = g_object_new (T_TYPE_DOUBLE, NULL);return d;
}

测试

完成对TInt、TDouble类型的定义,现在,看看这两个类型在应用程序中应该如何使用。

#include <glib-object.h>
#include "tnumber.h"
#include "tint.h"
#include "tdouble.h"static void
notify_cb (GObject *gobject, GParamSpec *pspec, gpointer user_data) {const char *name;int i;double d;name = g_param_spec_get_name (pspec);if (T_IS_INT (gobject) && strcmp (name, "value") == 0) {g_object_get (T_INT (gobject), "value", &i, NULL);g_print ("Property \"%s\" is set to %d.\n", name, i);} else if (T_IS_DOUBLE (gobject) && strcmp (name, "value") == 0) {g_object_get (T_DOUBLE (gobject), "value", &d, NULL);g_print ("Property \"%s\" is set to %lf.\n", name, d);}
}int
main (int argc, char **argv) {TNumber *i, *d, *num;char *si, *sd, *snum;i = T_NUMBER (t_int_new ());d = T_NUMBER (t_double_new ());g_signal_connect (G_OBJECT (i), "notify::value", G_CALLBACK (notify_cb), NULL);g_signal_connect (G_OBJECT (d), "notify::value", G_CALLBACK (notify_cb), NULL);g_object_set (T_INT (i), "value", 100, NULL);g_object_set (T_DOUBLE (d), "value", 12.345, NULL);num = t_number_add (i, d);si = t_number_to_s (i);sd = t_number_to_s (d);snum = t_number_to_s (num);g_print ("%s + %s is %s.\n", si, sd, snum);g_object_unref (num);g_free (snum);num = t_number_add (d, i);snum = t_number_to_s (num);g_print ("%s + %s is %s.\n", sd, si, snum);g_object_unref (num);g_free (sd);g_free (snum);g_object_set (T_DOUBLE (d), "value", 0.0, NULL);sd = t_number_to_s (d);if ((num = t_number_div(i, d)) != NULL) {snum = t_number_to_s (num);g_print ("%s / %s is %s.\n", si, sd, snum);g_object_unref (num);g_free (snum);}g_object_unref (i);g_object_unref (d);g_free (si);g_free (sd);return 0;
}

程序运行结果如下:

Property "value" is set to 100.
Property "value" is set to 12.345000.
100 + 12.345000 is 112.
12.345000 + 100 is 112.345000.
Property "value" is set to 0.000000.Error: division by zero.

有趣的是,两次加法的结果不同。这是为什么呢?TInt与TDouble都对TNumber 的add 函数进行了覆盖,那么当使用两个不同类型的数据调用加法函数时,应该使用哪个类型的add函数呢,这就取决于t_number_add的实现了。

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

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

相关文章

Linux内核中的锁

不同的锁&#xff0c;作用对象是不一样的&#xff0c;也就是作用域不一样下面分别是作用于临界区、CPU、内存、cache 的各种锁的归纳&#xff1a; 补充&#xff1a;cache是一种缓存&#xff0c;包含硬件缓存&#xff08;CPU缓存&#xff09;以及软件缓存&#xff08;网页缓存&a…

Cocos2d-x 4.0 工程首次建立与编译(Mac m1)

Mac m1芯片下将cocos2d-x升级至4.0版本后&#xff0c;官方剔除了不同平台的工程以及变更了编译方式&#xff0c;直接使用cmake构建&#xff0c;需要做一些前置的准备工作。 环境准备&#xff1a; 项 版本 备注 MacOS10.3 or laterpython2.7.16(建议>2.7.10)cmake3.29.3Do…

自动驾驶场景下TCP协议参数优化调整案例分享

RTT 往返时间&#xff0c;从tcp协议栈决定发包&#xff0c;到收到回包的时间。 包含本地驱动&#xff0c;网卡硬件&#xff0c;网线&#xff0c;交换机&#xff0c;收包方处理的耗时。需注意如果开了delayed ack&#xff0c;协议栈未做特殊处理&#xff08;默认没做&#xff…

C++中字符字面量的使用细节

C中字符字面量的使用细节 如何在 C中书写字符字面值:将字符用单引号括起&#xff0c;如M(注意&#xff0c;示例中没有使用双引号。C对字符用单引号&#xff0c;对字符串使用双引号。cout对象能够处理这两种情况&#xff0c;但正如第4章将讨论的&#xff0c;这两者有天壤之别)。…

【项目管理知识】挣值管理的概念与计算

【项目管理知识】挣值管理的概念与计算 四个基础概念&#xff1a;BAC、AC、PV、EV四个绩效概念&#xff1a;CV、CPI、SV、SPI四个延伸概念&#xff1a;ETC、EAC、非典型偏差、典型偏差挣值管理-非典型偏差挣值管理-典型偏差 完工尚需绩效指数&#xff1a;TCPI 四个基础概念&…

探索交互的本质:从指令到界面的演进与Linux基础指令的深入剖析

目录 1.指令 vs 界面//选读 1.1交互的需求 满足需求的第一阶段-指令 满足需求的第二阶段-界面 1.2 指令 和 界面交互 区别 2.操作系统介绍 2.1 举例说明 驱动软件层 2.2 为什么要有操作系统&#xff1f; 0x03 为什么要进行指令操作&#xff1f; 3.Linux基本指令 l…

模型量化 剪枝bevfusion

量化 剪枝 shared mem 只在block内共享&#xff0c;device glob mem能够所有线程共享

从多线程设计模式到对 CompletableFuture 的应用

大家好&#xff0c;我是 方圆。最近在开发 延保服务 频道页时&#xff0c;为了提高查询效率&#xff0c;使用到了多线程技术。为了对多线程方案设计有更加充分的了解&#xff0c;在业余时间读完了《图解 Java 多线程设计模式》这本书&#xff0c;觉得收获良多。本篇文章将介绍其…

重塑IT审计的未来:数智化审计赋能平台的创新与实践

重塑IT审计的未来&#xff1a;数智化审计赋能平台的创新与实践 一、当前企业开展IT审计面临的挑战 随着信息技术的快速发展、企业数字化转型的持续深入&#xff0c;以及网络安全合规要求的不断增强&#xff0c;企业开展新型IT审计重要性越来越突出&#xff0c;但实施难度却越来…

859.亲密字符串

给你两个字符串 s 和 goal &#xff0c;只要我们可以通过交换 s 中的两个字母得到与 goal 相等的结果&#xff0c;就返回 true &#xff1b;否则返回 false 。 交换字母的定义是&#xff1a;取两个下标 i 和 j &#xff08;下标从 0 开始&#xff09;且满足 i ! j &#xff0c…

【数据融合】基于卡尔曼滤波实现GPS-IMU数据融合附matlab代码

下面是一个简单的示例代码&#xff0c;用于基于卡尔曼滤波实现GPS-IMU数据融合的MATLAB实现&#xff1a; matlab % 初始化卡尔曼滤波器参数 dt 0.1; % 时间步长 A [1 dt; 0 1]; % 状态转移矩阵 B [0.5*dt^2; dt]; % 输入控制矩阵 H [1 0]; % 观测矩阵 Q eye(2); % 状态噪…

Python闯LeetCode--第2题:两数相加

Problem: 2. 两数相加 文章目录 思路解题方法复杂度Code 思路 看到这一题&#xff0c;第一反应是分别遍历两个链表&#xff0c;用列表来存储数&#xff0c;先把两个数的值拿到&#xff0c;转为具体数之后进行相加&#xff0c;再把两数相加值转为列表&#xff0c;再转为链表返回…

统计信号处理基础 习题解答10-16

题目&#xff1a; 对于例10.1&#xff0c;证明由观察数据得到的信息是&#xff1a; 解答&#xff1a; 基于习题10-15的结论&#xff0c;&#xff0c;那么&#xff1a; 而根据习题10-15的结论&#xff1a; 此条件概率也是高斯分布&#xff0c;即&#xff1a; 根据相同的计算&a…

国密SM2JS加密后端解密

1.前端加密 前端加密开源库 sm-crypto 1.1 传统web,下载 sm-crypto 进行打包为 dist/sm2.js 相关打包命令 npm install --save sm-crypto npm install npm run prepublish在web页面引用打包后的文件 <script type"text/javascript" src"<%path %>…

一文带你搞清楚AI领域的高频术语!RAG、Agent、知识库、向量数据库、知识图谱、Prompt...都是在讲啥?

随着AI人工智能技术的不断发展&#xff0c;一些领域有关的概念和缩写总是出现在各种文章里&#xff0c;像是Prompt Engineering、Agent 智能体、知识库、向量数据库、RAG 以及知识图谱等等&#xff0c;但是这些技术和概念也的的确确在AI大模型的发展中扮演着至关重要的角色。这…

使用 3D 图形 API 在 C# 中将 PLY 转换为 OBJ

OBJ和PLY是一些广泛使用的 3D 文件格式&#xff0c;易于编写和读取。这篇博文演示了如何以编程方式在 C# 中将 PLY 转换为 OBJ。此外&#xff0c;它还介绍了一种用于 3D 文件格式转换的在线3D 转换器。是的&#xff0c;Aspose.3D for .NET为程序员和非程序员提供了此功能来执行…

Day52 代码随想录打卡|二叉树篇---二叉搜索树中的众数

题目&#xff08;leecode T501&#xff09;&#xff1a; 给你一个含重复值的二叉搜索树&#xff08;BST&#xff09;的根节点 root &#xff0c;找出并返回 BST 中的所有 众数&#xff08;即&#xff0c;出现频率最高的元素&#xff09;。 如果树中有不止一个众数&#xff0c…

【深度学习】基于EANet模型的图像识别和分类技术

1.引言 1.1.EANet模型简介 EANet&#xff08;External Attention Transformer&#xff09;是一种深度学习模型&#xff0c;它结合了Transformer架构和外部注意力机制&#xff0c;特别适用于图像分类等计算机视觉任务。以下是关于EANet的详细解释&#xff1a; 1.1.1 定义与背…

微信小程序写一个录音机

微信小程序写一个录音机 代码&#xff1a; wxml: <view class"container"><view class"duration">{{duration}}</view><button bindtap"startRecord" class"btn">开始录音</button><button bindta…

Node.js版本管理工具-NVM

在开发 Node.js 项目时&#xff0c;经常会遇到需要切换不同版本的 Node.js 的情况。为了方便管理和切换各个版本&#xff0c;我们可以使用一些 Node.js 版本管理工具。 Node Version Manager&#xff1a;简称NVM&#xff0c;最流行的 Node.js 版本管理工具之一。它允许我们在同…