《TCP/IP网络编程》阅读笔记--基于UDP的服务器端/客户端

这篇具有很好参考价值的文章主要介绍了《TCP/IP网络编程》阅读笔记--基于UDP的服务器端/客户端。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

目录

1--TCP和UDP的主要区别

2--基于 UDP 的数据 I/O 函数

3--基于 UDP 的回声服务器端/客户端

4--UDP客户端Socket的地址分配

5--UDP存在数据边界

6--UDP已连接与未连接的设置


1--TCP和UDP的主要区别

① TCP 提供的是可靠数据传输服务,而 UDP 提供的是不可靠数据传输服务;

② UDP 在结构上比 TCP 更简洁,其不会发送 ACK 应答消息,也不会给数据包分配类似 SEQ 的序号;

③ 流控制是区分 UDP 和 TCP 的最重要标志,UDP 比 TCP 速度更快;

④ TCP 比 UCP 慢的主要两个原因:一是 TCP 在收发数据前后需要进行连接设置和清除过程;二是 TCP 在收发数据过程中为保证可靠性而添加的流控制;

⑤ TCP Socket 是一对一的关系,因此当需要为多个客户端提供服务时,就需要创建多个 TCP Socket,而 UDP 无论在服务器端还是客户端都只需要 1 个 Socket;

2--基于 UDP 的数据 I/O 函数

#include <sys/socket.h>
ssize_t sendto(int sock, void *buff, size_t nbytes, int flags, struct sockaddr *to, socklen_t addrlen); 
// 成功时返回传输的字节数,失败时返回 -1
// sock 表示传输数据的 UDP Socket 的文件描述符
// buff 表示保存待传输数据的缓冲地址值
// nbytes 表示待传输的数据长度,以字节为长度
// flags 可选项参数,若没有则传递 0
// to 存有目标地址信息的 sockaddr 结构体变量的地址值
// addrlen 传递给参数 to 的地址值结构体变量长度
#include <sys/socket.h>
ssize_t recvfrom(int sock, void *buff, size_t nbytes, int flags, struct sockaddr* from, socklen_t *addrlen);

// 成功时返回接收的字节数,失败时返回 -1
// sock 表示用于接收数据的 UDP socket的文件描述符
// nbytes 保存接收数据的缓冲地址值
// flags 可选项参数,若没有则传入 0
// from 存有发送端地址信息的 sockaddr 结构体变量的地址值
// addrlen 保存参数 from 的结构体变量长度的变量地址值

3--基于 UDP 的回声服务器端/客户端

服务器端:

// gcc uecho_server.c -o uecho_server
// ./uecho_server 9190

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>

#define BUF_SIZE 30

void error_handling(char *message){
    fputs(message, stderr);
    fputc('\n', stderr);
    exit(1);
}

int main(int argc, char *argv[]){
    int serv_sock;
    char message[BUF_SIZE];
    int str_len;
    socklen_t clnt_adr_sz;
    struct sockaddr_in serv_adr, clnt_adr;

    if(argc != 2){
        printf("Usage : %s <port>\n", argv[0]);
        exit(1);
    }

    serv_sock = socket(PF_INET, SOCK_DGRAM, 0);
    if(serv_sock == -1){
        error_handling("UDP socket creation error");
    }

    memset(&serv_adr, 0, sizeof(serv_adr));
    serv_adr.sin_family = AF_INET;
    serv_adr.sin_addr.s_addr = htonl(INADDR_ANY);
    serv_adr.sin_port = htons(atoi(argv[1]));

    if(bind(serv_sock, (struct sockaddr*)&serv_adr, sizeof(serv_adr)) == -1){
        error_handling("bind() error");
    }

    while(1){
        clnt_adr_sz = sizeof(clnt_adr);
        str_len = recvfrom(serv_sock, message, BUF_SIZE, 0, (struct sockaddr*)&clnt_adr, &clnt_adr_sz);
        sendto(serv_sock, message, str_len, 0, (struct sockaddr*)&clnt_adr, clnt_adr_sz);
    }
    close(serv_sock);
    return 0;
}

客户端:

// gcc uecho_client.c -o uecho_client
// ./uecho_client 127.0.0.1 9190

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>

#define BUF_SIZE 30

void error_handling(char *message){
    fputs(message, stderr);
    fputc('\n', stderr);
    exit(1);
}

int main(int argc, char *argv[]){
    int sock;
    char message[BUF_SIZE];
    int str_len;
    socklen_t adr_sz;
    struct sockaddr_in serv_adr, from_adr;

    if(argc != 3){
        printf("Usage : %s <IP> <port>\n", argv[0]);
        exit(1);
    }

    sock = socket(PF_INET, SOCK_DGRAM, 0);
    if(sock == -1){
        error_handling("socket() error");
    }

    memset(&serv_adr, 0, sizeof(serv_adr));
    serv_adr.sin_family = AF_INET;
    serv_adr.sin_addr.s_addr = inet_addr(argv[1]);
    serv_adr.sin_port = htons(atoi(argv[2]));

    while(1){
        fputs("Insert message(q to quit): ", stdout);
        fgets(message, sizeof(message), stdin);

        if(!strcmp(message, "q\n") || !strcmp(message, "Q\n")){
            break;
        }

        sendto(sock, message, strlen(message), 0, (struct sockaddr*)&serv_adr, sizeof(serv_adr));
        adr_sz = sizeof(from_adr);
        str_len = recvfrom(sock, message, BUF_SIZE, 0, (struct sockaddr*)&from_adr, &adr_sz);
        message[str_len] = 0;
        printf("Message from server: %s", message);
    }
    close(sock);
    return 0;
}

运行结果:

《TCP/IP网络编程》阅读笔记--基于UDP的服务器端/客户端,网络编程笔记,tcp/ip

4--UDP客户端Socket的地址分配

        TCP 客户端调用 connect() 函数自动完成 IP 和端口的分配;

        UDP 客户端调用 sento() 函数自动完成 IP 和端口的分配;

5--UDP存在数据边界

        TCP 数据传输不存在数据边界,而 UDP 数据传输存在数据边界,因此输入函数(recvfrom)和输出函数(sendto)的调用次数应该相同;

6--UDP已连接与未连接的设置

        UDP Socket 默认是未连接状态,因此每次传输数据都需要带上目标地址(IP、Port),这种属于未连接 Socket;

        类似于 TCP Socket,UDP同样可以调用 connect() 函数来设置目标地址,从而创建已连接的 UDP Socket;

        已连接的 UDP Socket 由于指定了收发对象,因此不仅可以使用 sendto、recvfrom 函数,还可以使用 write 和 read 函数进行通信;

        基于已连接 UDP 的回声客户端如下:

// gcc uecho_con_client.c -o uecho_con_client
// ./uecho_con_client 127.0.0.1 9190

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>

#define BUF_SIZE 30

void error_handling(char *message){
    fputs(message, stderr);
    fputc('\n', stderr);
    exit(1);
}

int main(int argc, char *argv[]){
    int sock;
    char message[BUF_SIZE];
    int str_len;
    socklen_t adr_sz;
    struct sockaddr_in serv_adr, from_adr;

    if(argc != 3){
        printf("Usage : %s <IP> <port>\n", argv[0]);
        exit(1);
    }

    sock = socket(PF_INET, SOCK_DGRAM, 0);
    if(sock == -1){
        error_handling("socket() error");
    }

    memset(&serv_adr, 0, sizeof(serv_adr));
    serv_adr.sin_family = AF_INET;
    serv_adr.sin_addr.s_addr = inet_addr(argv[1]);
    serv_adr.sin_port = htons(atoi(argv[2]));

    connect(sock, (struct sockaddr*)&serv_adr, sizeof(serv_adr));

    while(1){
        fputs("Insert message(q to quit): ", stdout);
        fgets(message, sizeof(message), stdin);

        if(!strcmp(message, "q\n") || !strcmp(message, "Q\n")){
            break;
        }

        write(sock, message, strlen(message));
        str_len = read(sock, message, sizeof(message) - 1);
        message[str_len] = 0;
        printf("Message from server: %s", message);
    }
    close(sock);
    return 0;
}

《TCP/IP网络编程》阅读笔记--基于UDP的服务器端/客户端,网络编程笔记,tcp/ip文章来源地址https://www.toymoban.com/news/detail-708311.html

到了这里,关于《TCP/IP网络编程》阅读笔记--基于UDP的服务器端/客户端的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 《TCP/IP网络编程》阅读笔记--多播与广播

    目录 1--多播 2--多播代码实例 3--广播 4--广播代码实例         多播方式的数据传输是基于 UDP 完成的,多播数据包的格式与 UDP 数据包相同;         多播与 UDP 的区别:UDP 数据传输以单一目标进行,多播数据同时传递到加入(注册)特定组的大量主机; 多播的数据传输

    2024年02月07日
    浏览(49)
  • 《TCP/IP网络编程》阅读笔记--并发多进程服务端的使用

    目录 1--并发服务器端 2--进程 2-1--进程的相关概念 2-2--fork()创建进程 2-3--僵尸进程 2-4--wait()和waitpid()销毁僵尸进程 3--信号处理 3-1--signal()函数 3-2--sigaction()函数 3--3--利用信号处理技术消灭僵尸进程 4--基于多任务的并发服务器 5--分割 TCP 的 I/O 程序 并发服务器端主要有以下三类

    2024年02月09日
    浏览(34)
  • 《TCP/IP网络编程》阅读笔记--getsockopt和setsockopt的使用

    目录 1--Socket的多种可选项 2--getsocketopt() 3--setsockopt() 4--代码实例         Socket 拥有多种可选项,其可分为 SOL_SOCKET 层,IPPROTO_IP 层和IPPROTO_TCP 层等,一般通过 getsocketopt() 和 setsockopt() 函数进行获取和设置; ① 基于  getsockopt() 函数,利用设置协议层为 SOL_SOCKET 和 SO_TYPE 可选项

    2024年02月09日
    浏览(33)
  • 《TCP/IP网络编程》阅读笔记--Timewait状态和Nagle算法

            对于服务器端/客户端,当一端结束连接时,会向另一端发送 FIN 消息;两端的在经过四次挥手过程后,其 Socket 不会马上消除,而是会处于一个 Time-wait 状态的阶段,此时 Socket 拥有的 端口号并没有得到释放 ,因此 不能使用相同的端口号 ;         只有先断开连接

    2024年02月09日
    浏览(39)
  • 网络编程——TCP/IP协议族(IP协议、TCP协议和UDP协议……)

    1、IP协议简介 IP协议又称 网际协议 特指为实现在一个相互连接的网络系统上从源地址到目的地传输数据包(互联网数据包)所提供必要功能的协议,是网络层中的协议。 2、特点 不可靠 :它不能保证IP数据包能成功地到达它的目的地,仅提供尽力而为的传输服务 无连接 :IP 并不

    2024年02月13日
    浏览(59)
  • 【Java学习笔记】 68 - 网络——TCP编程、UDP编程

    https://github.com/yinhai1114/Java_Learning_Code/tree/main/IDEA_Chapter21/src 目录 项目代码 网络 一、网络相关概念 1.网络通讯 2.网络 3.IP地址 4.域名 5.端口号 6.网络通讯协议 TCP协议:传输控制协议 UDP协议: 二、InetAddress类 1.相关方法 三、Socket 1.基本介绍 2.TCP网络通信编程 基本介绍 应用案例

    2024年02月04日
    浏览(47)
  • 基于TCP、UDP网络编程

    应用层 : 应用程序拿到数据怎么用 传输层 : 负责关注传输过程中起点和终点 网络层 :负责整个传输过程中的路线规划 数据链路层 :负责相邻节点之间的数据传输 物理层 : 基础设施,硬件设备 程序员写网络程序,主要编写的应用层代码,真正要发这个数据,调用下层协

    2024年02月07日
    浏览(44)
  • Java网络编程之IP,端口号,通信协议(UDP,TCP)

    ① C/S :客户端/服务器 在用户本地需要下载安装客户端程序,在远程有一个服务器端程序。 优点:画面精美,用户体验好 缺点:用户需要下载更新 ② B/S :浏览器/服务器 只需要一个浏览器,用户通过指定网址访问对应的服务器。 优点:不需要开发客户端,只需要页面+服务

    2024年02月03日
    浏览(69)
  • 【网络编程】网络编程概念,socket套接字,基于UDP和TCP的网络编程

    前言: 大家好,我是 良辰丫 ,今天我们一起来学习网络编程,网络编程的基本概念,认识套接字,UDP与TCP编程.💞💞💞 🧑个人主页:良辰针不戳 📖所属专栏:javaEE初阶 🍎励志语句:生活也许会让我们遍体鳞伤,但最终这些伤口会成为我们一辈子的财富。 💦期待大家三连,关注

    2023年04月20日
    浏览(48)
  • 基于UDP/TCP的网络通信编程实现

    红色是心中永不褪色的赤诚 操作系统为网络编程提供了 Socket api , Socket是基于TCP/IP协议的网络通信的基本单元, 基于Socket的网络程序开发就是 网络编程. 由于直接与应用层联系的是传输层, 所以针对应用层协议(TCP, UDP), Shocket提供了三种套接字, 分别是 流套接字(使用TCP) , 数据报

    2024年02月08日
    浏览(47)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包