如何从 python 调用用 C 编写的 GObject 类的方法?

Posted

技术标签:

【中文标题】如何从 python 调用用 C 编写的 GObject 类的方法?【英文标题】:How can I call methods on a GObject class written in C from python? 【发布时间】:2021-11-23 00:57:06 【问题描述】:

我正在尝试在 C 中创建一个 GObject 类并以某种方式对其进行注释,以便我可以使用 Python 中的类 - 但我认为缺少一些东西,因为我遇到了我无法理解的奇怪错误。任何帮助将不胜感激!

如果我尝试从其他 C 代码中使用它,该类将按预期工作,并且我可以生成 .gir.typelib 文件,据我所知,XML 看起来是正确的。

当我尝试从 Python 创建类的实例时,似乎没有调用对象的 _class_init_init 函数,如果我尝试调用方法,则会出现异常:

from gi.repository import Bop
foo = Bop.FooObj()
foo.get_number_of_tap_dancers()
# TypeError: expected GObject but got <gi.repository.Bop.FooObj object at 0x104670100>

我试图为我的类 (bop_foo_obj_new) 显式创建一个构造函数,如果我从 Python 调用它,我可以看到 _class_init_init get 函数都被调用,但是 Python 解释器会出现段错误:

Bop.FooObj.new()
# bop_foo_obj_new() called!!!!
# bop_foo_obj_class_init() called!!!!
# bop_foo_obj_init() called!!!!
#
# ** (process:25468): CRITICAL **: 12:55:19.436: pygobject_register_wrapper: assertion # 'PyObject_TypeCheck(self, &PyGObject_Type)' failed
# ./wat.sh: line 4: 25468 Segmentation fault: 11  GI_TYPELIB_PATH=build LD_LIBRARY_PATH=build python ./wat.py

这是我要开始工作的代码:

bop.h:

#pragma once
#include <glib-object.h>

G_BEGIN_DECLS

G_DECLARE_DERIVABLE_TYPE(BopFooObj, bop_foo_obj, BOP, FOO_OBJ, GObject);

struct _BopFooObjClass 
  GObjectClass parent;
;

BopFooObj *bop_foo_obj_new();
guint32 bop_foo_obj_get_number_of_tap_dancers(BopFooObj *self);

G_END_DECLS

bop.c:

#include "bop.h"
#include <stdio.h>

typedef struct 
  guint tap_dancer_cnt;
 BopFooObjPrivate;

G_DEFINE_TYPE_WITH_PRIVATE(BopFooObj, bop_foo_obj, G_TYPE_OBJECT);

static void bop_foo_obj_class_init(BopFooObjClass *klass) 
  fprintf(stderr, "bop_foo_obj_class_init() called!!!!\n");


static void bop_foo_obj_init(BopFooObj *self) 
  fprintf(stderr, "bop_foo_obj_init() called!!!!\n");
  BopFooObjPrivate *priv = bop_foo_obj_get_instance_private(self);
  priv->tap_dancer_cnt = 9999;


/**
 * bop_foo_obj_new:(constructor):
 * Returns:(transfer full):
 */
BopFooObj *bop_foo_obj_new() 
  fprintf(stderr, "bop_foo_obj_new() called!!!!\n");
  return BOP_FOO_OBJ(g_object_new(bop_foo_obj_get_type(), NULL));


/**
 * bop_foo_obj_get_number_of_tap_dancers:
 * @self: the #BopFooObj instance
 */
guint32 bop_foo_obj_get_number_of_tap_dancers(BopFooObj *self) 
  BopFooObjPrivate *priv = bop_foo_obj_get_instance_private(self);
  return priv->tap_dancer_cnt;

生成的 .gir 文件:

<?xml version="1.0"?>
<!-- This file was automatically generated from C sources - DO NOT EDIT!
To affect the contents of this file, edit the original C definitions,
and/or use gtk-doc annotations.  -->
<repository version="1.2"
            xmlns="http://www.gtk.org/introspection/core/1.0"
            xmlns:c="http://www.gtk.org/introspection/c/1.0"
            xmlns:glib="http://www.gtk.org/introspection/glib/1.0">
  <namespace name="Bop"
             version="0.1"
             shared-library="libwat.dylib"
             c:identifier-prefixes="Bop"
             c:symbol-prefixes="bop">
    <class name="FooObj"
           c:symbol-prefix="foo_obj"
           c:type="BopFooObj"
           glib:type-name="BopFooObj"
           glib:get-type="bop_foo_obj_get_type"
           glib:type-struct="FooObjClass">
      <source-position filename="../bop.h" line="10"/>
      <constructor name="new" c:identifier="bop_foo_obj_new">
        <source-position filename="../bop.h" line="12"/>
        <return-value transfer-ownership="full">
          <type name="FooObj" c:type="BopFooObj*"/>
        </return-value>
      </constructor>
      <method name="get_number_of_tap_dancers"
              c:identifier="bop_foo_obj_get_number_of_tap_dancers">
        <source-position filename="../bop.h" line="13"/>
        <return-value transfer-ownership="none">
          <type name="guint32" c:type="guint32"/>
        </return-value>
        <parameters>
          <instance-parameter name="self" transfer-ownership="none">
            <doc xml:space="preserve"
                 filename="../bop.c"
                 line="31">the #BopFooObj instance</doc>
            <type name="FooObj" c:type="BopFooObj*"/>
          </instance-parameter>
        </parameters>
      </method>
      <field name="parent_instance" introspectable="0">
        <type c:type="GObject"/>
      </field>
    </class>
    <record name="FooObjClass"
            c:type="BopFooObjClass"
            glib:is-gtype-struct-for="FooObj">
      <source-position filename="../bop.h" line="10"/>
      <field name="parent" introspectable="0">
        <type c:type="GObjectClass"/>
      </field>
    </record>
  </namespace>
</repository>

【问题讨论】:

您能否为该课程发布一份 GIR XML 副本?通常问题可以通过查看来诊断。还请发布您正在使用的编译命令/构建脚本的副本,以及您用于运行 Python 代码的命令。 当然!我更新了帖子 @PhilipWithnall 我使用的构建工具 i Meson,感谢您的评论,我开始更仔细地研究它实际创建的内容。 XML 在我的类上丢失了`parent="GObject.Object"`,结果证明这是因为我必须将includes: 'GObject-2.0' 添加到我的meson.build 文件的gnome.generate_gir 部分。谢谢:-) 然后解决了吗?如果是这样,太好了!如果您将您的解决方案写出来(并将其标记为问题的解决方案),对其他人会有所帮助。 【参考方案1】:

事实证明,这是由我使用的构建系统 Meson 调用 g-ir-scanner 的方式引起的。我注意到它使用的是--extra-library=GObject-2.0,而我能找到的关于这个主题的少数教程使用--include=GObject-2.0 调用g-ir-scanner

我将includes: 'GObject-2.0' 添加到我的meson.build 文件的gnome.generate_gir 部分并解决了问题:

最终的 meson.build 文件如下所示:

project('wat', 'c', default_options : ['c_std=c17'])

gnome = import('gnome')
deps = [
  dependency('gobject-2.0'),
  dependency('gobject-introspection-1.0')
]

headers = [
  'bop.h'
]
sources = [
  'bop.c'
]

wat_lib = library('wat',
    sources,
    dependencies: deps,
    install: true,
)

gnome.generate_gir(
    wat_lib,
    namespace: 'Bop',
    nsversion: '0.1',
    sources: headers + sources,
    dependencies: deps,
    install: true,
    fatal_warnings: true,
    includes: 'GObject-2.0',   # this line fixed the issue.
)

【讨论】:

以上是关于如何从 python 调用用 C 编写的 GObject 类的方法?的主要内容,如果未能解决你的问题,请参考以下文章

使用参数从 Python 调用 C/C++ 代码

python绑定,它是如何工作的?

如何在 MicroPython 中从 C 调用 python 函数

如何将 Cython 生成的模块从 python 导入 C/C++ 主文件? (用 C/C++ 编程)[关闭]

我可以从c ++调用用cupy代码编写的cuda代码吗?

RIDE如何调用自定义Python文件中的函数