计算机网络——GBN协议实现

devtools/2024/9/22 16:08:46/

实验目的

编程模拟实现GBN可靠传输软件

实验内容

C++ 程序模拟实现Go-Back-N可靠数据传输,需要编写一个发送端程序和一个测试端程序来模拟传输过程

具体流程

1. 编写发送端程序,调用库实现socket连接,然后主要实现滑动窗口,接收ACK确认帧和超时重传的功能
2. 编写接收端,调用监听一个端口,因为在本地实验,要模拟丢包的过程,使用随机数模拟出百分之十的丢包率。编写丢包前后不同的ACK确认帧逻辑

关键代码

发送端,先建立连接,然后接收用户在命令行的输入,获取要发送的消息和滑动窗口。然后开始按照GBN协议发送,发送完一帧后就滑动窗口往下。发送完一帧的同时还会开启一个计时器,计时器延时一秒钟,没有收到ACK就会从超时的位置开始重传

    asio::io_service io_service;tcp::socket socket(io_service);socket.connect(tcp::endpoint(asio::ip::address::from_string("127.0.0.1"), 12345));std::string message;std::cout << "Input message: ";std::getline(std::cin, message);int window_size;std::cout << "Window size: ";std::cin >> window_size;std::vector<char> window(window_size);int base = 0;int next_seq_num = 0;asio::steady_timer timer(io_service);timer.expires_from_now(std::chrono::seconds(1));while (base < message.size()) {while (next_seq_num < base + window_size && next_seq_num < message.size()) {window[next_seq_num % window_size] = message[next_seq_num];char seq_num_char = static_cast<char>(next_seq_num);asio::write(socket, asio::buffer(&seq_num_char, 1));asio::write(socket, asio::buffer(&window[next_seq_num % window_size], 1)); std::cout << "Seq_num: " << next_seq_num << "  Send: " << window[next_seq_num % window_size] << std::endl;next_seq_num++;}char ack;asio::async_read(socket, asio::buffer(&ack, 1), [&](const asio::error_code& error, std::size_t length) {if (!error) {for (int i = 0; i < 25; i++) {std::cout << " ";}std::cout << "Receive ACK: " << static_cast<int>(ack) << std::endl;base = ack + 1;if (base == next_seq_num) {timer.cancel();} else {timer.expires_from_now(std::chrono::seconds(1));}} else {std::cout << "Error reading ACK: " << error.message() << std::endl;}});timer.async_wait([&](const asio::error_code& error) {if (!error) {std::cout << "Timeout resend from: " << base << std::endl;for (int i = base; i < next_seq_num; i++) {char seq_num_char = static_cast<char>(i);asio::write(socket, asio::buffer(&seq_num_char, 1));asio::write(socket, asio::buffer(&window[i % window_size], 1));}}});io_service.run();io_service.reset();}

接收端,先开启端口的监听,然后开始接收消息,中间模拟随机百分之十的丢包率,如果发生丢包只会发送重复最后收到的连续的帧。还有一段处理错误的代码,接收结束消息,安全退出 

    asio::io_service io_service;tcp::acceptor acceptor(io_service, tcp::endpoint(tcp::v4(), 12345));tcp::socket socket(io_service);acceptor.accept(socket);srand(time(0));int expected_seq_num = -1;while (true) {char seq_num_char;char data;asio::error_code error;asio::read(socket, asio::buffer(&seq_num_char, 1), error);  // read sequence numberasio::read(socket, asio::buffer(&data, 1), error);  // read dataif (error) {if (error == asio::error::eof) {std::cout << "\nAll message have been received, connection closed" << std::endl;} else {std::cout << "Error reading: " << error.message() << std::endl;}break;}int seq_num = static_cast<int>(seq_num_char);if (expected_seq_num == -1) {int random_num = rand() % 10;if (random_num == 0) {std::cout << "Loss Seq_num: " << seq_num<< std::endl;expected_seq_num = seq_num;continue;}std::cout << "Seq_num: " << seq_num << "  Receive: " << static_cast<char>(data) << std::endl;}else {std::cout << "Seq_num: " << seq_num << "  Receive: " << static_cast<char>(data) << std::endl;if (seq_num == expected_seq_num) {expected_seq_num = -1;}else {seq_num = expected_seq_num - 1;}}asio::write(socket, asio::buffer(&seq_num, 1));for (int i = 0; i < 25; i++) {std::cout << " ";}std::cout << "Send ACK: " << seq_num << std::endl;}

运行示例

因为有两个程序运行,所以我们打开两个终端。先编译链接生成发送端和接收端的程序

 先再一个终端中运行接收端,程序,监听对应的端口。然后再在另外一个终端中运行发送端程序,输入要发送的信息,和滑动窗口的大小。这里我设置的要发送的信息为“thiisatestmessage”,滑动窗口大小为5

然后程序就会开开始模拟,可以看到发送端一次性快速发送完了窗口中的内容,但是接收端在一个一个接收到只之后才会发送对应帧的ACK。可以看到我们在模拟的过程中丢失了4号和15号两个帧,这两个帧丢失后的ACK都只会重复发送丢失帧之前的最后一个ACK,然后发送端超时之后就会从对应的位置开始重新发送

完整代码

BJTU_CS_Learning/computernetwork at main · JJLi0427/BJTU_CS_Learning (github.com) 


http://www.ppmy.cn/devtools/4111.html

相关文章

神经网络压缩图像

简介 典型的压缩管道由四个组件组成&#xff1a; 编码&#xff1a;输入图像 x x x通过编码器函数 ε \varepsilon ε&#xff0c;将其转换为潜在表示 z z z。 量化&#xff1a;截断 z z z以丢弃一些不重要的信息 熵编码&#xff1a;使用某种形式的熵编码&#xff08;例如&…

Tomcat下载配置地址

IntelliJ IDEA是一个强大的集成开发环境&#xff0c;能够大大简化Java应用程序的开发和部署过程。而Tomcat作为一个流行的Java Web服务器&#xff0c;其与IntelliJ IDEA的整合能够提供便捷的开发环境&#xff0c;让开发人员更专注于代码的创作与优化。 在配置IntelliJ IDEA以使…

docker最新版安装

docker安装 检查系统版本即卸载旧docker安装docker依赖工具及底层依赖、仓库源安装dockerdocker阿里云镜像资源站参考 检查系统版本即卸载旧docker # 查看操作系统的发行版号 uname -r# 查看系统版本 cat /etc/redhat-release# 卸载旧版本docker(如已安装过) yum remove docke…

Jackson 2.x 系列【25】Spring Boot 集成之起步依赖、自动配置

有道无术&#xff0c;术尚可求&#xff0c;有术无道&#xff0c;止于术。 本系列Jackson 版本 2.17.0 本系列Spring Boot 版本 3.2.4 源码地址&#xff1a;https://gitee.com/pearl-organization/study-jaskson-demo 文章目录 1. 前言2. 起步依赖3. 自动配置3.1 JacksonPrope…

力扣:141. 环形链表

力扣&#xff1a;141. 环形链表 给你一个链表的头节点 head &#xff0c;判断链表中是否有环。 如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;则链表中存在环。 为了表示给定链表中的环&#xff0c;评测系统内部使用整数 pos 来表示链表尾…

【MySQL面试题pro版-13】

MySQL是一个关系型数据库管理系统&#xff0c;由瑞典 MySQL AB 公司开发&#xff0c;属于 Oracle 旗下产品。MySQL是最流行的关系型数据库管理系统之一&#xff0c;在 WEB 应用方面&#xff0c;MySQL是最好的RDBMS (Relational Database Management System&#xff0c;关系数据…

负载均衡的原理及算法简介

负载均衡&#xff08;Load Balancing&#xff09;是一种用于在多台服务器之间分配网络流量的技术&#xff0c;旨在优化系统资源利用率、提高服务可用性、增强系统的伸缩性和容错能力。其基本原理是将来自客户端的请求分散到一个服务器集群中的各个服务器上&#xff0c;而不是让…

什么是bs架构和cs架构

bs架构是在浏览器就能访问到的系统&#xff0c;不需要用户去更新和安装app cs架构是需要下载一个app客户端&#xff0c;每次升级都需要更新app