Socket编程实践(12) --UDP编程基础

简介: UDP特点   无连接,面向数据报(基于消息,不会粘包)的数据传输服务;   不可靠(可能会丢包, 乱序, 重复), 但因此一般情况下UDP更加高效;UDP客户/服务器模型  U...

UDP特点

   无连接,面向数据报(基于消息,不会粘包)的数据传输服务;

   不可靠(可能会丢包, 乱序, 重复), 但因此一般情况下UDP更加高效;


UDP客户/服务器模型

 

 

UDP-API使用

#include <sys/types.h>
#include <sys/socket.h>
ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags,
                 struct sockaddr *src_addr, socklen_t *addrlen);
ssize_t sendto(int sockfd, const void *buf, size_t len, int flags,
               const struct sockaddr *dest_addr, socklen_t addrlen);

/**实践: 实现一个基于UDP的echo回声server/client**/
//server端代码
void echoServer(int sockfd);
int main()
{
    int sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (sockfd == -1)
        err_exit("socket error");

    struct sockaddr_in servAddr;
    servAddr.sin_family = AF_INET;
    servAddr.sin_addr.s_addr = htonl(INADDR_ANY);
    servAddr.sin_port = htons(8001);
    if (bind(sockfd, (const struct sockaddr *)&servAddr, sizeof(servAddr)) == -1)
        err_exit("bind error");

    echoServer(sockfd);
}
void echoServer(int sockfd)
{
    char buf[BUFSIZ];
    ssize_t recvBytes = 0;
    struct sockaddr_in clientAddr;
    socklen_t addrLen;
    while (true)
    {
        memset(buf, 0, sizeof(buf));
        addrLen = sizeof(clientAddr);
        memset(&clientAddr, 0, addrLen);
        recvBytes = recvfrom(sockfd, buf, sizeof(buf), 0,
                             (struct sockaddr *)&clientAddr, &addrLen);
        //如果recvBytes=0, 并不代表对端连接关闭, 因为UDP是无连接的
        if (recvBytes < 0)
        {
            if (errno == EINTR)
                continue;
            else
                err_exit("recvfrom error");
        }

        cout << buf ;
        if (sendto(sockfd, buf, recvBytes, 0,
                   (const struct sockaddr *)&clientAddr, addrLen) == -1)
            err_exit("sendto error");
    }
}
/**client端代码**/
void echoClient(int sockfd);
int main()
{
    int sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (sockfd == -1)
        err_exit("socket error");
    echoClient(sockfd);
    cout << "Client exiting..." << endl;
}
void echoClient(int sockfd)
{
    struct sockaddr_in servAddr;
    servAddr.sin_family = AF_INET;
    servAddr.sin_addr.s_addr = inet_addr("127.0.0.1");
    servAddr.sin_port = htons(8001);
    char buf[BUFSIZ] = {0};
    while (fgets(buf, sizeof(buf), stdin) != NULL)
    {
        if (sendto(sockfd, buf, strlen(buf), 0,
                   (const struct sockaddr *)&servAddr, sizeof(servAddr)) == -1)
            err_exit("sendto error");
        memset(buf, 0, sizeof(buf));
        int recvBytes = recvfrom(sockfd, buf, sizeof(buf), 0, NULL, NULL);
        if (recvBytes == -1)
        {
            if (errno == EINTR)
                continue;
            else
                err_exit("recvfrom error");
        }
        cout << buf ;
        memset(buf, 0, sizeof(buf));
    }
}

实践解析:编译运行server,在两个终端里各开一个client与server交互,可以看到server具有并发服务的能力。用<Ctrl+C>关闭server,然后再运行server,此时client还能和server联系上。和前面TCP程序的运行结果相比较,我们可以体会无连接的含义。udp 协议来说,server与client 的界限更模糊了,只要知道对等方地址(ip和port) 都可以主动发数据。

 

UDP编程注意事项

   1.UDP报文可能会丢失(超时重传)、重复、乱序(维护一个序号)

   2.UDP缺乏流量控制:当缓冲区写满以后,由于UDP没有流量控制机制,因此会覆盖缓冲区。

   3.UDP协议数据报文截断:如果对端发送的UDP数据报大于本地接收缓冲区,报文可能被截断,后面的部分会丢失(而不是像我们想象的下一次能够接收到)。

   4.recvfrom可以返回0,并不代表连接关闭,因为UDP是无连接的, 代表发送端没有发送任何数据[sendto可以发送数据0包(只含有UDP+IP首部40B)]。

   5.ICMP异步错误

      观察现象:使用上例,关闭UDP服务端,启动客户端,从键盘接受数据后,再发送数据。如果recvfrom中flags标志为0, 且client端没有调用connect的情况下, UDP客户端阻塞在recvfrom位置(见测试代码3);

      说明:

         1)UDP发送报文的时,只把数据copy到发送缓冲区。在服务器没有起来的情况下,可以发送成功

         2)所谓ICMP异步错误是指:发送的报文的时候,没有错误,接受报文recvfrom的时候,回收到ICMP应答.

         3)异步的错误,无法返回未连接的套接字, 因此如果上例我们调用了connect, 是可以收到该异步ICMP报文的;

   6.UDP调用connect

      1)UDP调用connet,并没有三次握手,只是维护了一个(和对等方的)状态信息, 因此我们可以看到即使server没有开启, client端的connect依然还可以正确返回的!(测试代码如测试代码2)

      2)一但调用connect, 发送可以使用send/write, 接收可以使用recv/read函数(见测试代码3)

   7.UDP外出接口的确定:

      假设客户端有多个IP地址,由connect /sendto 函数提供的远程地址的参数,系统会选择一个合适的出口,比如Server的IP是192.168.2.10, 而客户端现在的IP有 192.168.1.32 和 192.168.2.75 那么会自动选择192.168.2.75 这个IP出去。

/**测试1: 测试注意点3, UDP报文截断, recvfrom返回-1, errno值为EAGAIN**/
int main()
{
    int sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (sockfd == -1)
        err_exit("socket error");

    struct sockaddr_in servAddr;
    servAddr.sin_family = AF_INET;
    servAddr.sin_addr.s_addr = htonl(INADDR_ANY);
    servAddr.sin_port = htons(8001);
    if (bind(sockfd, (const struct sockaddr *)&servAddr, sizeof(servAddr)) == -1)
        err_exit("bind error");
    //给自己发送数据
    if (sendto(sockfd, "ABCDE", 5, 0,
               (const struct sockaddr *)&servAddr, sizeof(servAddr)) == -1)
        err_exit("sendto error");

    for (int i = 0; i < 5; ++i)
    {
        char ch;
        int recvBytes =  recvfrom(sockfd, &ch, 1, MSG_DONTWAIT, NULL, NULL);
        if (recvBytes == -1)
        {
            if (errno == EINTR)
                continue;
            else if (errno == EAGAIN)
                err_exit("recvfrom error");
        }
        else
            cout << "char = " << ch << ", recvBytes = " << recvBytes << endl;
    }
}
/**测试2:将client端echoClient函数的代码改造如下, 注意是在server端尚未开启时执行该程序**/
void echoClient(int sockfd)
{
    struct sockaddr_in servAddr;
    servAddr.sin_family = AF_INET;
    servAddr.sin_addr.s_addr = inet_addr("127.0.0.1");
    servAddr.sin_port = htons(8001);
    // UDP client端调用connect
    if (connect(sockfd, (const struct sockaddr *)&servAddr, sizeof(servAddr)) == -1)
        err_exit("connect error");

    char buf[BUFSIZ] = {0};
    while (fgets(buf, sizeof(buf), stdin) != NULL)
    {
        if (sendto(sockfd, buf, strlen(buf), 0,
                   (const struct sockaddr *)&servAddr, sizeof(servAddr)) == -1)
            err_exit("sendto error");
        memset(buf, 0, sizeof(buf));
        int recvBytes = recvfrom(sockfd, buf, sizeof(buf), 0, NULL, NULL);
        if (recvBytes == -1)
            err_exit("recvfrom error");
        cout << buf ;
        memset(buf, 0, sizeof(buf));
    }
}
/**测试3: client端在调用connect之后调用send, 而不是send**/
void echoClient(int sockfd)
{
    struct sockaddr_in servAddr;
    servAddr.sin_family = AF_INET;
    servAddr.sin_addr.s_addr = inet_addr("127.0.0.1");
    servAddr.sin_port = htons(8001);
    // UDP client端调用connect
    if (connect(sockfd, (const struct sockaddr *)&servAddr, sizeof(servAddr)) == -1)
        err_exit("connect error");

    char buf[BUFSIZ] = {0};
    while (fgets(buf, sizeof(buf), stdin) != NULL)
    {
        if (send(sockfd, buf, strlen(buf), 0) == -1)
            err_exit("send error");
        memset(buf, 0, sizeof(buf));
        int recvBytes = recv(sockfd, buf, sizeof(buf), 0);
        if (recvBytes == -1)
            err_exit("recv error");
        cout << buf ;
        memset(buf, 0, sizeof(buf));
    }
}


目录
相关文章
|
6天前
|
网络协议 Linux
TCP 和 UDP 的 Socket 调用
【9月更文挑战第6天】
|
2月前
|
网络协议 开发者 Python
深度探索Python Socket编程:从理论到实践,进阶篇带你领略网络编程的魅力!
【7月更文挑战第25天】在网络编程中, Python Socket编程因灵活性强而广受青睐。本文采用问答形式深入探讨其进阶技巧。**问题一**: Socket编程基于TCP/IP,通过创建Socket对象实现通信,支持客户端和服务器间的数据交换。**问题二**: 提升并发处理能力的方法包括多线程(适用于I/O密集型任务)、多进程(绕过GIL限制)和异步IO(asyncio)。**问题三**: 提供了一个使用asyncio库实现的异步Socket服务器示例,展示如何接收及响应客户端消息。通过这些内容,希望能激发读者对网络编程的兴趣并引导进一步探索。
27 4
|
2月前
|
网络协议 Python
网络世界的建筑师:Python Socket编程基础与进阶,构建你的网络帝国!
【7月更文挑战第26天】在网络的数字宇宙中,Python Socket编程是开启网络世界大门的钥匙。本指南将引领你从基础到实战,成为网络世界的建筑师。
49 2
|
2月前
|
网络协议 程序员 视频直播
|
2月前
|
消息中间件 网络协议 网络安全
Python Socket编程:打造你的专属网络通道,基础篇与进阶篇一网打尽!
【7月更文挑战第26天】在网络编程领域,Python以简洁语法和强大库支持成为构建应用的首选。Socket编程为核心,实现计算机间的数据交换。
50 1
|
2月前
|
安全 网络协议 网络安全
Python Socket编程大揭秘:从菜鸟到黑客的进阶之路,你准备好了吗?
【7月更文挑战第27天】Python Socket编程是网络开发的关键技能,它开启从简单数据传输到复杂应用的大门。Socket作为网络通信的基础,通过Python的`socket`模块可轻松实现跨网通信。
33 0
|
2月前
|
网络协议 Python
告别网络编程迷雾!Python Socket编程基础与实战,让你秒变网络达人!
【7月更文挑战第27天】在网络编程的广阔天地中,Socket编程常被视为一道难关。但用Python这把钥匙,我们可以轻松入门。Socket作为网络通信的基石,在Python中通过`socket`模块封装了底层细节,简化了开发过程。以下是一个基本的TCP服务器与客户端的示例,展示了如何建立连接、收发数据及关闭连接。为了应对实际场景中的并发需求,我们还可以借助多线程技术来提升服务器处理能力。掌握了这些基础知识后,你将逐步揭开网络编程的神秘面纱,踏上编程高手之路!
29 0
|
3月前
|
缓存 监控 Java
Java Socket编程最佳实践:优化客户端-服务器通信性能
【6月更文挑战第21天】Java Socket编程优化涉及识别性能瓶颈,如网络延迟和CPU计算。使用非阻塞I/O(NIO)和多路复用技术提升并发处理能力,减少线程上下文切换。缓存利用可减少I/O操作,异步I/O(AIO)进一步提高效率。持续监控系统性能是关键。通过实践这些策略,开发者能构建高效稳定的通信系统。
105 1
|
3月前
|
Java
Java Socket编程与多线程:提升客户端-服务器通信的并发性能
【6月更文挑战第21天】Java网络编程中,Socket结合多线程提升并发性能,服务器对每个客户端连接启动新线程处理,如示例所示,实现每个客户端的独立操作。多线程利用多核处理器能力,避免串行等待,提升响应速度。防止死锁需减少共享资源,统一锁定顺序,使用超时和重试策略。使用synchronized、ReentrantLock等维持数据一致性。多线程带来性能提升的同时,也伴随复杂性和挑战。
74 0
|
3月前
|
安全 Java 网络安全
Java Socket编程教程:构建安全可靠的客户端-服务器通信
【6月更文挑战第21天】构建安全的Java Socket通信涉及SSL/TLS加密、异常处理和重连策略。示例中,`SecureServer`使用SSLServerSocketFactory创建加密连接,而`ReliableClient`展示异常捕获与自动重连。理解安全意识,如防数据截获和中间人攻击,是首要步骤。通过良好的编程实践,确保网络应用在复杂环境中稳定且安全。
74 0