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

embedded/2024/11/12 2:46:13/

实验目的

编程模拟实现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/embedded/18839.html

相关文章

使用 Dify 和 Moonshot API 构建你的 AI 工作流(一):让不 AI 的应用 AI 化

有了之前的文章铺垫&#xff0c;这篇文章开始&#xff0c;我们聊聊如何折腾 AI 工作流&#xff0c;把不 AI 的应用&#xff0c;“AI 起来”。 写在前面 上个月&#xff0c;我们聊过了《使用 Dify 和 AWS Bedrock 玩转 Anthropic Claude 3》&#xff0c;里面介绍了如何使用交互…

Android Studio Iguana | 2023.2.1配置优化

一. 前言 本篇文章记录最新版本的Android Studio的配置优化&#xff0c;写这篇文章的是由于电脑中的AS工具更新版本覆盖安装后&#xff0c;AS会经常卡死&#xff0c;Debug的时候也经常莫名其妙的断掉&#xff0c;非常影响工作效率&#xff0c;所以重新把配置环境整理一下&#…

微信小程序关于主包大小不能超过1.5MB的问题

常规的解决办法有以下几种 1、把资源文件改成远程服务器的&#xff0c;比如png这些 2、进入如图的分析页面&#xff0c;能明确知道你哪个插件包太大&#xff0c;我这里之前echart的包就1mb&#xff0c;现在给他缩减到了500kb的样子 3、解决vant等npm包太大的问题&#xff0c…

git 分支重命名 使用IDEA进行操作

当前项目下分支下执行 git push origin -d 旧分支名称 git push origin 新分支名称 tips: 本地分支重命名。如果分支尚未推送到远程&#xff0c;可以使用命令“git branch -m 旧名称 新名称”来重命名本地分支。 远程分支重命名。首先&#xff0c;重命名本地分支&#xff08;与…

浅谈菊风实时音视频 (RTC)与实时操作系统 (RTOS) 在智能硬件领域应用

近年来&#xff0c;菊风通过实时音视频赋能智能手表、智能门禁、智能门锁/门铃、智能眼镜等数十种智能硬件&#xff0c;与一众合作伙伴共同探索在IoT智能硬件领域的不同场景应用&#xff0c;积累了丰富的实践经验。在智能硬件中&#xff0c;RTOS因其轻量化的系统内核&#xff0…

沉浸式推理乐趣:体验线上剧本杀小程序的魅力

在这个信息爆炸的时代&#xff0c;人们的娱乐方式也在不断地推陈出新。其中&#xff0c;线上剧本杀小程序以其独特的沉浸式推理乐趣&#xff0c;成为了许多人的新宠。它不仅让我们在闲暇之余享受到了推理的快乐&#xff0c;更让我们在虚拟的世界里感受到了人性的复杂与多彩。 线…

25计算机考研院校数据分析 | 南京大学

南京大学&#xff08;Nanjing University&#xff09;&#xff0c;简称“南大”&#xff0c;是中华人民共和国教育部直属、中央直管副部级建制的全国重点大学&#xff0c;国家首批“双一流”、“211工程”、“985工程”重点建设高校&#xff0c;入选首批“珠峰计划”、“111计划…

免费的单片机物联网MQTT平台选择

目的是多设备接入中控&#xff0c;平台只做转发。 选择巴法云&#xff1a;巴法科技&巴法云-巴法设备云-巴法物联网云平台 clientId是私钥uid&#xff1a; 多设备 clientId 填同一个 uid 都是可以的。平台应该是加了后缀区分。 支持自定义topic&#xff0c;操作简单&#x…