首页 > 解决方案 > 从函数返回的 GtkWidget* 导致分段错误

问题描述

我在问为什么如果我决定在另一个文件中调用一个函数,如果使用返回的 GtkWidget* 会导致分段错误(如示例中的强制转换、gtk 函数......)

这里主要我运行 gtk_init 并从另一个文件中定义的函数中尝试 GTK_IS_WIDGET

主程序

#include <gtk/gtk.h>
#include <glib.h>

int main(){
  gtk_init(NULL, NULL);
  GTK_IS_WIDGET(new_widget());
}

在另一个文件中,我创建了一个新的 GtkWidget 并使用函数 GTK_IS_WIDGET,然后我打印一个“OK”以了解执行是否到达这一点

extern_file.c

#include<gtk/gtk.h>
GtkWidget* new_widget(){
  GtkWidget *new = gtk_window_new(GTK_WINDOW_TOPLEVEL);
  GTK_IS_WIDGET(new);
  g_print("OK\n");
  return new;
}

我以这种方式编译这两个文件,gcc -o main.o main.c esterno.c $(pkg-config --cflags --libs glib-2.0 gtk+-2.0) 我得到了很多警告,(我把它放在那里)但没有错误。

main.c:6:17: warning: implicit declaration of function ‘new_widget’ [-Wimplicit-function-declaration]
   GTK_IS_WIDGET(new_widget());
                 ^
/usr/include/glib-2.0/gobject/gtype.h:2238:44: note: in definition of macro ‘_G_TYPE_CIT’
   GTypeInstance *__inst = (GTypeInstance*) ip; GType __t = gt; gboolean __r; \
                                            ^~
/usr/include/gtk-2.0/gtk/gtkwidget.h:139:35: note: in expansion of macro ‘G_TYPE_CHECK_INSTANCE_TYPE’
 #define GTK_IS_WIDGET(widget)    (G_TYPE_CHECK_INSTANCE_TYPE ((widget), GTK_TYPE_WIDGET))
                                   ^~~~~~~~~~~~~~~~~~~~~~~~~~
main.c:6:3: note: in expansion of macro ‘GTK_IS_WIDGET’
   GTK_IS_WIDGET(new_widget());
   ^~~~~~~~~~~~~
/usr/include/glib-2.0/gobject/gtype.h:2238:27: warning: cast to pointer from integer of different size [-Wint-to-pointer-cast]
   GTypeInstance *__inst = (GTypeInstance*) ip; GType __t = gt; gboolean __r; \
                           ^
/usr/include/glib-2.0/gobject/gtype.h:495:66: note: in expansion of macro ‘_G_TYPE_CIT’
 #define G_TYPE_CHECK_INSTANCE_TYPE(instance, g_type)            (_G_TYPE_CIT ((instance), (g_type)))
                                                                  ^~~~~~~~~~~
/usr/include/gtk-2.0/gtk/gtkwidget.h:139:35: note: in expansion of macro ‘G_TYPE_CHECK_INSTANCE_TYPE’
 #define GTK_IS_WIDGET(widget)    (G_TYPE_CHECK_INSTANCE_TYPE ((widget), GTK_TYPE_WIDGET))
                                   ^~~~~~~~~~~~~~~~~~~~~~~~~~
main.c:6:3: note: in expansion of macro ‘GTK_IS_WIDGET’
   GTK_IS_WIDGET(new_widget());

如果我运行文件 ( ./main.o),我会收到“OK”消息,确认函数内部的 GTK_IS_WIDGET 工作正常,但是当在 main 中调用 GTK_IS_WIDGET 时出现分段错误。

如果我在 main 中声明函数一切正常,但这不是一个真正的解决方案。

提前致谢

标签: csegmentation-faultgtkgtk2

解决方案


我收到很多警告,(我把它放在那里)但没有错误。

对于 C 语言来说,这是一种危险的心态,任何警告都可能是致命的。我建议-Wall在开发时使用 with gcc(并且只在特定情况下放宽)。

在这种情况下,编译器会这样警告你:

main.c:6:17: warning: implicit declaration of function ‘new_widget’

int new_widget()在 C 中,这意味着在调用您的函数时使用默认签名 。您的实际函数签名当然与此不匹配,因此开始发生坏事。

您可以通过添加包含正确函数签名的头文件并在 main.c 中包含该头文件来解决此问题。


推荐阅读