Glib/Gio 异步或线程 UDP 服务器
Posted
技术标签:
【中文标题】Glib/Gio 异步或线程 UDP 服务器【英文标题】:Glib/Gio Asynchronous or Threaded UDP Server 【发布时间】:2015-05-14 17:51:59 【问题描述】:我目前有一个同步 UDP 应用程序接收消息。
代码:
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <glib.h>
#include <gio/gio.h>
int main(argc,argv)
int argc;
char ** argv;
char buf[256], *ptr, sep[] = "| ";
GError * error = NULL;
GSocket * socket;
GSocketAddress *gsockAddr, *gfromAddr;
guint16 udp_port = 1500;
//Creates socket udp ipv4
socket = g_socket_new(G_SOCKET_FAMILY_IPV4,
G_SOCKET_TYPE_DATAGRAM,
G_SOCKET_PROTOCOL_UDP,
&error);
g_assert(error == NULL);
if (socket == NULL)
g_print("ERROR");
exit(1);
//sockaddr struct like
gsockAddr = G_SOCKET_ADDRESS(g_inet_socket_address_new(g_inet_address_new_any(G_SOCKET_FAMILY_IPV4), udp_port));
if(gsockAddr == NULL)
g_error("Error socket\n");
exit(1);
//
if (g_socket_bind (socket, gsockAddr, TRUE, NULL) == FALSE)
g_print("Error bind\n");
exit(1);
int bytes = g_socket_receive_from (socket,
&gfromAddr,
buf,
255,
NULL,
&error);
if (bytes == -1)
g_warning ("Failed to receive from socket: %s", error->message);
g_error_free (error);
return TRUE;
g_message("Server receive: %s", buf);
guint16 port = g_inet_socket_address_get_port(G_INET_SOCKET_ADDRESS(gfromAddr));
g_print("...from %s(%d)\n",g_inet_address_to_string(g_inet_socket_address_get_address(G_INET_SO CKET_ADDRESS(gfromAddr))), (int) port);
exit(0);
所以,我想让接收操作,非阻塞而不是阻塞。我想让它成为异步的或/和线程化的,以便同时我可以执行与我想要开发的应用程序相关的其他操作。
但我没有成功地做到我想要的。我尝试使用 GLib IO 通道,但无法使其正常工作。进程正在等待,但这只是因为主循环(我无法远程登录应用程序)。
代码:
#include <gio/gio.h>
#include <glib.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#define BLOCK_SIZE 1024
static gboolean
gio_read_socket (GIOChannel *channel,
GIOCondition condition,
gpointer data)
char buf[1024];
gsize bytes_read;
GError *error = NULL;
if (condition & G_IO_HUP) return FALSE; /* this channel is done */
g_io_channel_read_chars (channel, buf, sizeof (buf), &bytes_read,
&error);
g_assert (error == NULL);
buf[bytes_read] = '\0';
g_print ("%s", buf);
return TRUE;
int
main (int argc, char **argv)
GSocket * s_udp;
GError *err = NULL;
guint16 udp_port = 5556;
s_udp = g_socket_new(G_SOCKET_FAMILY_IPV4,
G_SOCKET_TYPE_DATAGRAM,
G_SOCKET_PROTOCOL_UDP,
&err);
g_assert(err == NULL);
if (s_udp == NULL)
g_print("ERROR");
exit(1);
g_socket_bind(s_udp,
G_SOCKET_ADDRESS(g_inet_socket_address_new(g_inet_address_new_any(G_SOCKET_FAMILY_IPV4), udp_port)),
TRUE,
&err);
g_assert(err == NULL);
int fd = g_socket_get_fd(s_udp);
GIOChannel* channel = g_io_channel_unix_new(fd);
guint source = g_io_add_watch(channel, G_IO_IN,
(GIOFunc) gio_read_socket, NULL);
g_io_channel_unref(channel);
GMainLoop *loop = g_main_loop_new(NULL, FALSE);
g_main_loop_run(loop);
g_main_loop_unref(loop);
我是 GLib/Gio 的初学者,我认为我在 IO 通道方面做错了。我想将它作为一个事件添加到主循环中,以便我可以使用我的回调函数。也许有更简单的方法来做到这一点。
此外,我有一个正在工作的 TCP 异步和线程服务器,但我没有找到如何对 UDP 做同样的事情(使用 GThreadedSocketService 并创建一个套接字侦听器,然后将服务添加到主循环。简单馅饼与 TCP)。
您知道如何进行吗?如果你知道怎么做但只使用基本的 API 套接字,我仍然接受它!谢谢。
【问题讨论】:
您的代码中有一些额外的空格,例如G_SOCKET_FAM ILY_IPV4
是的,确实,谢谢,我的复制/粘贴失败了,但第一个代码块完全有效。我“只是”想让它成为非阻塞的。
我对 glib 了解不多,但我怀疑您需要将频道设置为无缓冲 (developer.gnome.org/glib/stable/…)。您几乎可以肯定必须将文件描述符置于非阻塞模式(fcntl(soc, F_SETFL, O_NONBLOCK)
,尽管从技术上讲,您应该获取标志,设置 O_NONBLOCK 标志并设置新标志)。
另见***.com/questions/9513327/…
感谢您的回答,如果我使用基本的 API 套接字和 ioctl 函数,我可能会这样做,但是使用 Glib/Gio,我突然感到困惑。无论如何,我确实找到了如何让它工作,见下文。再次感谢您的关心。
【参考方案1】:
我想通了。
我确实是个初学者。因为,当我想测试我的 udp 应用程序(第二个代码块)时,我使用 telnet 连接到它并尝试发送消息。但是,我们当然不能 telnet udp 应用程序...
所以我尝试使用一个简单的 udp 发送器(顺便说一下,我使用了 Glib/Gio)而不是 telnet,它可以正常工作,完全无阻塞且可重用。我确实做了一些改变,但基本上是一样的。我放了一个空闲函数来向你展示它是如何无阻塞的,这是否可以帮助某人。
我的简单 Glib/Gio UDP 应用,非阻塞:
#include <gio/gio.h>
#include <glib.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#define BLOCK_SIZE 1024
static gboolean
gio_read_socket (GIOChannel *channel,
GIOCondition condition,
gpointer data)
char buf[1024];
gsize bytes_read;
GError *error = NULL;
if (condition & G_IO_HUP) return FALSE; /* this channel is done */
g_io_channel_read_chars (channel, buf, sizeof (buf), &bytes_read,
&error);
g_assert (error == NULL);
buf[bytes_read] = '\0';
g_print ("%s", buf);
int *a = data;
*a = *a + 1;
return TRUE;
gboolean
idleCpt (gpointer user_data)
int *a = user_data;
g_print("%d\n", *a);
sleep(1);
return TRUE;
int
main (int argc, char **argv)
GSocket * s_udp;
GError *err = NULL;
int idIdle = -1, dataI = 0;
guint16 udp_port = 1505;
GSocketAddress * gsockAddr = G_SOCKET_ADDRESS(g_inet_socket_address_new(g_inet_address_new_any(G_SOCKET_FAMILY_IPV4), udp_port));
s_udp = g_socket_new(G_SOCKET_FAMILY_IPV4,
G_SOCKET_TYPE_DATAGRAM,
G_SOCKET_PROTOCOL_UDP,
&err);
g_assert(err == NULL);
if (s_udp == NULL)
g_print("ERREUR");
exit(1);
if (g_socket_bind (s_udp, gsockAddr, TRUE, NULL) == FALSE)
g_print("Erreur bind\n");
exit(1);
g_assert(err == NULL);
int fd = g_socket_get_fd(s_udp);
GIOChannel* channel = g_io_channel_unix_new(fd);
guint source = g_io_add_watch(channel, G_IO_IN,
(GIOFunc) gio_read_socket, &dataI);
g_io_channel_unref(channel);
GMainLoop *loop = g_main_loop_new(NULL, FALSE);
idIdle = g_idle_add(idleCpt, &dataI);
g_main_loop_run(loop);
代码并不完美,有很多优化要做,但我认为我们可以做一些好的事情。如果您想查看我的 udp 发件人,请询问。
【讨论】:
顺便说一句,你可以使用nc -u
或socat udp-connect
以上是关于Glib/Gio 异步或线程 UDP 服务器的主要内容,如果未能解决你的问题,请参考以下文章
在没有 GNOME 的情况下使用 glib gio GVolumeMonitor