我正在使用 C 和 unix 低级套接字开发聊天应用程序。我已经成功制作了控制台版本,但我想为应用程序制作一个 GUI。我想将 GTK 用于 GUI。我的问题是如何“同步”套接字和 GUI。因为我必须调用 gtk_main() 作为最后一个 GTK 语句,而应用程序本身就是一个无限循环。当消息进入时如何更新 GUI?
2 回答
您面临的问题是一次有多个事件系统但只有一个线程。Gtk+ 带有自己的事件处理程序,最终归结为select()
将在任何用户输入或其他 gtk 事件时唤醒。您自己想使用自己的事件处理来处理网络,这通常包括select()
在您的套接字上或在阻塞模式下使用套接字。
一种解决方案是将您的事件集成到 Gtk+ 的事件循环中。
您可以让 Gtk+ 监视/select()
您的套接字并在它们的状态更改时调用特定函数(数据可读)。请参阅http://developer.gnome.org/glib/2.30/glib-The-Main-Event-Loop.html上的“创建新的源类型”部分
另一种解决方案是使用 Gtk+ 网络功能。
通常,您不想对套接字做一些特别的事情,以至于它不容易用 Glib IO 通道包装。见http://developer.gnome.org/glib/2.30/glib-IO-Channels.html
第三种解决方案是启动第二个线程来处理您的网络,例如使用 posix 线程或 Gtk+ 线程功能。
将 GUI 与应用程序的工作部分分开通常是一个好主意。但是对于聊天应用程序,它可能不会比其他解决方案带来任何好处。见http://developer.gnome.org/glib/2.30/glib-Threads.html
这是 python 中的一个示例,pygobject
用于GLib.IOChannel
在 gtk 主事件循环中添加监视。
一只手表用于收听新的连接。另一个用于接收数据。
这改编自这个 pygtk 示例:http ://rox.sourceforge.net/desktop/node/413.html
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk, GLib;
from socket import socket
def listener(io, cond, sock):
conn = sock.accept()[0]
GLib.io_add_watch(GLib.IOChannel(conn.fileno()),0,GLib.IOCondition.IN, handler, conn)
return True
def handler(io, cond, sock):
print(sock.recv(1000))
return True
s = socket()
s.bind(('localhost', 50555))
s.listen()
GLib.io_add_watch(GLib.IOChannel(s.fileno()), 0, GLib.IOCondition.IN, listener, s)
Gtk.main()