c - GObject OOP 语法

标签 c oop gtk gobject

我正在寻找 GObject 备忘单,了解常见的 OOP 概念如何映射到 GObject 的设施。例如考虑:

AnyGObject *o;
o = anygobject_new();

现在,约定是什么......

  • 调用方法
  • 调用基类声明的方法
  • 调用类正在实现的接口(interface)声明的方法
  • 调用类方法
  • 调用基类声明的类方法
  • 调用虚拟方法
  • 转换为基类
  • 转换为派生类
  • 转换为类正在实现的接口(interface)
  • 测试对象是否属于特定类型
  • ...

澄清: GObject Reference ManualGObject HowTo详细解释如何创建一个新类(类结构、对象结构、私有(private)结构、各种宏、约定)。将这些设施结合在一起就可以实现 OOP。然而似乎没有关于如何一致使用它们的教程。

最佳答案

此答案假设您正在使用 C。其他(通常是面向对象的)语言具有特殊的绑定(bind),以使使用 GObject 看起来更加自然。

如果您使用过 GTK+,那么您已经完成了该列表中的大部分内容。

GObject 方法本身不是成员(有一个 vtable,但它仅用于在首次创建类时在派生类中分配虚拟方法实现)。相反,GObject 中的所有方法都只是普通函数,通常(?)以方法名称前缀为前缀,并以 this 指针作为第一个参数。

例如C++方法

namespace Lib { // short for Library; to demonstrate GObject's idiomatic naming conventions
    class Foo {
    public:
        void Bar(int z);
    };
}

将是全局命名空间中的一个普通函数,声明为

void lib_foo_bar(LibFoo *foo, int z);

您可以直接调用它,就像任何其他 C 函数一样。

GObject 中的类派生是通过将父类的完整数据结构作为派生类数据结构的第一个成员来进行的。由于与 C 标准中很少讨论的子句有关的各种原因(可能还有 System V ABI 和 gcc、clang 甚至 Microsoft C 编译器的实现),这意味着指向派生类对象的指针等效于指向父类的指针!

因此,如果 LibBaz 派生自 LibFoo,那么您只需说

LibFoo *foobaz = (LibFoo *) baz;

反之亦然:

LibBaz *bazfoo = (LibBaz *) foo;

(后一种方法是 GTK+ 用于 GtkWidget 的方法;我不知道其他 GObject 库是否也做同样的事情。)

惯用的 GObject 声明包含一堆宏,使类型转换更加简洁,同时添加运行时类型安全检查。我们的 LibFoo 类将具有以下宏:

#define LIB_TYPE_FOO (lib_foo_get_type())
#define LIB_FOO(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj), LIB_TYPE_FOO, LibFoo))

有了这个,我们会说

LibFoo *foobaz = LIB_FOO(baz);
LibBaz *bazfoo = LIB_BAZ(foo);

如果 bazfoo 不是正确的转换类型,则会在标准错误中记录一条警告,您可以使用该错误进行中断并进行调查调试器。

(lib_foo_get_type() 函数(和 LIB_TYPE_FOO 整洁宏))很重要:它返回一个映射到 LibFoo 类型的数字 ID > 供将来引用。如果 LibFoo 没有这样的映射,它将创建映射、注册类型并创建虚拟方法映射。)

类似的宏允许类型检查:

#define LIB_IS_FOO(obj) (G_TYPE_CHECK_INSTANCE_TYPE((obj), LIB_TYPE_FOO))

这是一个可以在 if 语句中使用的简单表达式。

那么调用父类方法呢?好吧,如果我们将以上所有内容放在一起,我们就会得到答案:

lib_foo_parent_method(LIB_FOO(aLibBazInstance), params);

这同样适用于虚拟方法。虚拟方法是使用 GObject 近似 vtable 来实现的,并且对最终程序员是透明的。您要做的就是

lib_foo_virtual_method(LIB_FOO(whatever), params);

(如果您实际构建派生类本身,虚拟方法的方式就变得很重要。)

GObject 中没有静态方法,因为方法不像真正的面向对象语言那样与类紧密相关。只需在您的库中创建一个顶级方法即可:

void lib_something_common(params);

最后,以上所有内容都适用于接口(interface)。对于最终用户来说,界面的工作方式完全相同;他们使用相同的

void lib_iface_method(LibIface *iface, params);

方法调用方法、相同的转换规则以及相同的 LIB_IFACE()LIB_IS_IFACE() 辅助宏。

希望有帮助!任何进一步的解释都必须解释如何创建 GObject,为了简单起见,我试图将其保留在本答案的范围之外,但无论如何,了解它是一件有用的事情。

关于c - GObject OOP 语法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27823299/

相关文章:

c - 毕达哥拉斯三元组,不全等

返回指向抽象类的指针的 C++ 函数

c# - "protected"和 "virtual/override"之间的区别

python - 用于 python-evince 的简单 "Hello-World"程序

c - 从文件扩展名获取 Mime 类型

挑战: Calculate transfer rate via UDP socket

c++ - 如何计算精度为 E=0.0001 的序列 e^(-x) 之和?

c - movq 指令出现段错误?

Python oop : __init__ methods of class have often trivial assignments self. 值=值

Gtk+ 和 Vala : Inheriting gtk. TreeModel