按照自己的理解实现比特交换协议(alternating-bit protocol)
一开始的思路是想写两个程序的,发送端和接收端。但是后来想了下,因为是模拟的,所以不用这么麻烦,直接自己定制场景(比如说丢包阿,包出错阿,超时之类的)。
基本上是根据上图所写的,一个函数发包,一个函数接包(比如上图的wait_for_ack0),在这个发包函数里定制场景,在接包函数里处理场景。
说到处理场景,仔细看看wait for ack0上所写的:
翻译一下:
if ( rdt_rcv(rcvpkt) && (corrupt(rcvpkt)) || isACK(rcvpkt, ) ) // 如果有接受到包且包坏了, 或者接受到的ACK是1,那么就什么都不用做,因为会执行下面的超时函数
{
;// do nothing,静静地等待超时然后重新发包
}
if (timeout) // 如果超时,重新发包然后重新启动计时器
{
udt_send(sndpkt);
start_timer;
}
if ( rdt_rcv(rcvpkt) && notcorrupt(rcvpkt) && isACK(rcvpkt, ) ) // 只有凑齐接收到包且包正确且收到的回复正确,那么就可以停止计时了
{
stop timer;
}
因为这是停等协议(stop and wait),所以顺序是先发0包,再发1包,再发0包这样持续下去。
根据场景,我定义了这个一个数组:数据包丢失,确认包丢失,数据包出错,确认包出错,超时 5个场景。 丢失就会超时,出错就是验证校验和。
不过因为我懒,所以没有计算RTT之类的时间,校验和也忽略掉的。
这里给个思路,可以根据公式(大家可以看看计算机网路的书,上面有写算RTT的公式),只要你定义了数据包的大小,网速的话,根据包的个数是可以算出来RTT的。
下面是我自己定义的场景。
#define PACKAGE_LOSS 0
#define ACK_LOSS 1
#define PACK_CORRUPT 2
#define ACK_CORRUPT 3
#define TIME_OUT 4 int Flag[] = {}; // 0代表没有,1代表有
你在调用发包这个函数时就询问会出现哪种场景,我写得代码一次发包只能实现一种场景(不过根据场景的不同,有可能引发其他场景。比如丢包,这样自然就会引起超时了)。
在接收包的函数时,就定义解决场景,根据上图写。
下面给出完整代码(有一些地方可能会出错,但也弄出个大概,仅供参考)
#include <stdio.h>
#include <stdlib.h> #define DATA_LENGTH 20 struct Pkt
{
int seqnum;
int checksum;
char data[DATA_LENGTH];
}; #define PACKAGE_LOSS 0
#define ACK_LOSS 1
#define PACK_CORRUPT 2
#define ACK_CORRUPT 3
#define TIME_OUT 4 int Flag[] = {}; int nsimmax = ; // the number of message
int n_package = ;
struct Pkt *pkt_ptr = NULL; void init()
{
int i = ;
int j = ;
int k = ; printf("---- Stop and Wait Network Simulator Version 1.0 ----\n");
printf("Enter the number of messages to simulate: ");
scanf("%d", &nsimmax); pkt_ptr = (struct Pkt *)malloc(nsimmax * sizeof(struct Pkt)); // nsimmax packages
if (pkt_ptr == NULL)
{
perror("Malloc Error: ");
exit();
} // Fill some data in Pkt.data for (i = ; i < nsimmax; i++)
{
for (j = ; j < DATA_LENGTH; j++)
{
pkt_ptr[i].data[j] = + k;
} k++;
} } void Question(int *p)
{
int i = ; printf("--------------\n");
printf("0 - Nothing wrong\n");
printf("1 - Packet loss\n");
printf("2 - ACK loss\n");
printf("3 - Packet error\n");
printf("4 - ACK error\n");
printf("5 - Time out\n");
printf("--------------\n"); printf("Choice: ");
scanf("%d", &i); if (i != )
{
p[i - ] = ;
}
} void send()
{ if (Flag[] == )
{
printf("SEND: Package loss\n");
Flag[] = ; // time out
} else if (Flag[] == )
{
printf("SEND: ACK loss\n");
Flag[] = ; // time out
} else if (Flag[] == )
{
printf("SEND: Packet error\n");
} else if (Flag[] == )
{
printf("SEND: ACK error\n");
} else if (Flag[] == )
{
printf("SEND: Time out\n");
} else
{
printf("SEND: Nothing wrong\n");
} printf("\n"); } void rdt_send0()
{
Question(Flag); printf("SEND: Send package 0\n");
printf("SEND: Start timer\n\n"); send(); } void rdt_send1()
{
Question(Flag); printf("SEND: Send package 1\n");
printf("SEND: Start timer\n\n"); send(); } void waitACK0()
{
int i = ; while ()
{
if ((Flag[] == && Flag[] == ) || Flag[] == )
{
printf("-------------------\n");
if (Flag[] == ) // Send error package
{
printf("RECV: SEND NAK0\n");
} if (Flag[] == ) // Error ACK0 means ACK1
{
printf("RECV: Receive package 0\n"); printf("The data is: ");
for (i = ; i < DATA_LENGTH; i++)
printf("%c", pkt_ptr[n_package].data[i]);
printf("\n"); printf("RECV: SEND ACK1\n");
printf("SEND: ACK should be 0\n");
}
printf("-------------------\n"); Flag[] = ; // package error or error ACK can lead to time out
} if (Flag[] == ) // time out
{
printf("SEND: Time out\n");
printf("SEND: Resend package0...\n");
if (Flag[] == ) // package 0 loss
{
Flag[] = ;
} if (Flag[] == ) // ACK 0 loss
{
printf("RECV: Detetch the redundant, discard the package0\n");
Flag[] = ;
} if (Flag[] == ) // package 0 error
{
Flag[] = ;
} if (Flag[] == )
{
printf("RECV: Detetch the redundant, discard the package0\n");
Flag[] = ;
} printf("SEND: Start timer\n");
Flag[] = ;
} if (Flag[] == && Flag[] == && Flag[] == )
{
printf("SEND: Stop timer\n");
printf("-------------------\n");
printf("RECV: reveive package0\n");
printf("RECV: send ACK0\n"); printf("The data is: ");
for (i = ; i < DATA_LENGTH; i++)
printf("%c", pkt_ptr[n_package].data[i]);
printf("\n"); break;
}
}
} void waitACK1()
{
int i = ; while ()
{
if ((Flag[] == && Flag[] == ) || Flag[] == )
{
printf("-------------------\n");
if (Flag[] == ) // Send error package
{
printf("RECV: SEND NAK0\n");
} if (Flag[] == ) // Error ACK0 means ACK1
{
printf("RECV: Receive package 0\n"); printf("The data is: ");
for (i = ; i < DATA_LENGTH; i++)
printf("%c", pkt_ptr[n_package].data[i]);
printf("\n"); printf("RECV: SEND ACK1\n");
printf("SEND: ACK should be 0\n");
}
printf("-------------------\n"); Flag[] = ; // package error or error ACK can lead to time out
} if (Flag[] == ) // time out
{
printf("SEND: Time out\n");
printf("SEND: Resend package0...\n");
if (Flag[] == ) // package 1 loss
{
Flag[] = ;
} if (Flag[] == ) // ACK 1 loss
{
printf("RECV: Detetch the redundant, discard the package1\n");
Flag[] = ;
} if (Flag[] == ) // package 1 error
{
Flag[] = ;
} if (Flag[] == )
{
printf("RECV: Detetch the redundant, discard the package1\n");
Flag[] = ;
} printf("SEND: Start timer\n");
Flag[] = ;
} if (Flag[] == && Flag[] == && Flag[] == )
{
printf("SEND: Stop timer\n");
printf("-------------------\n");
printf("RECV: reveive package1\n");
printf("RECV: send ACK1\n"); printf("The data is: ");
for (i = ; i < DATA_LENGTH; i++)
printf("%c", pkt_ptr[n_package].data[i]);
printf("\n"); break;
}
}
} void run()
{
init();
while ()
{
printf("--------%d package ------------\n", n_package);
rdt_send0();
waitACK0();
n_package++;
if (n_package == nsimmax)
break; printf("--------%d package ------------\n", n_package);
rdt_send1();
waitACK1();
n_package++; if (n_package == nsimmax)
break;
} return; } int main(void)
{
run();
return ;
}
因为是停等协议,所以有两个send函数,有两个wait for ACK函数,都是差不多的,只是一个是发0包,接0包,一个是发1包,接1包。
下面给出运行结果:
按照自己的理解实现比特交换协议(alternating-bit protocol)的更多相关文章
- 社区观点 | 理解比原链MOV链上交换协议
去中心化交换协议的发展 从Bitshare,Stellar到以太坊上的Etherdelta,Bancor,0x协议,去中心化交换协议也经过了好几代发展和很多模式的探索,每一代都通过前面的协议的痛点来进 ...
- 自研DCI网络路由交换协议DCIP-白牌交换机时代的企业网络
一转眼从听华为3Com的路由交换课程到如今已经13年有余了,依稀记得第一节课的时候我带着老婆去听的课(老婆是日语系的.那时还是女朋友,并不懂网络,仅仅是跟着我去上课的).抢了个头排,讲师宋岩老师提问了 ...
- 简易远程消息交换协议SRMP
一.SRMP目标定位 经过十多年实战经验积累以及多方共同讨论,新生命团队(https://github.com/newlifex)制订了一种简单而又具有较好扩展性的RPC(Remote Procedu ...
- 理解分布式一致性:Paxos协议之Generalized Paxos & Byzantine Paxos
理解分布式一致性:Paxos协议之Generalized Paxos & Byzantine Paxos Generalized Paxos Byzantine Paxos Byzantine ...
- 理解分布式一致性:Paxos协议之Cheap Paxos & Fast Paxos
理解分布式一致性:Paxos协议之Cheap Paxos & Fast Paxos Cheap Paxos Message flow: Cheap Multi-Paxos Fast Paxos ...
- 理解分布式一致性:Paxos协议之Multi-Paxos
理解分布式一致性:Paxos协议之Multi-Paxos Multi-Paxos without failures Multi-Paxos when phase 1 can be skipped Mu ...
- 理解分布式一致性:Paxos协议之Basic Paxos
理解分布式一致性:Paxos协议之Basic Paxos 角色 Proposal Number & Agreed Value Basic Paxos Basic Paxos without f ...
- 理解分布式一致性:Raft协议
理解分布式一致性:Raft协议 什么是分布式一致性 Leader选举 日志复制流程 term选举周期 timeout 选举和选举timeout 选举分裂 日志复制和心跳timeout 在分布式系统中, ...
- 浏览器扩展系列————异步可插入协议(pluggable protocol)的实现
原文:浏览器扩展系列----异步可插入协议(pluggable protocol)的实现 IE中有很多我们比较熟悉的协议,如http,https,mailto,ftp等.当然你也可以实现自己定义的协议 ...
随机推荐
- MHA手动切换 原创2 (主参与复制)
monitor 执行下面命令后, --orig_master_is_new_slave :原主变为新从,即server1变成了slave1的从.slave2跟据app1.conf中配制也变成了slav ...
- 关于try...catch...finally中return的疑惑
原文:http://www.cnblogs.com/and_he/archive/2012/04/17/2453703.html 关于try...catch...finally里面的return一直是 ...
- 用count(*)还是count(列名) || Mysql中的count()与sum()区别
Mysql中的count()与sum()区别 首先创建个表说明问题 CREATE TABLE `result` ( `name` varchar(20) default NULL, `su ...
- Turn the pokers
Problem Description During summer vacation,Alice stay at home for a long time, with nothing to do. S ...
- cocos2d-x之jni使用(对接Android各种sdk)
游戏弄完了,要发布到各个平台,ios.Android是肯定少不了的,那么本文就来讲讲Android平台对接代理商付费sdk.各渠道.五大运营商.广告.分享.数据统计等等少不了的jni调用,接sdk真是 ...
- jQuery实例属性和方法
jQuery.fn = jQuery.prototype = { //添加实例属性和方法 jquery : 版本 constructor : 修正指向问题 init() : 初始化和参数 ...
- [改善Java代码]构造函数尽量简化
建议34: 构造函数尽量简化 我们知道在通过new关键字生成对象时必然会调用构造函数,构造函数的简繁情况会直接影响实例对象的创建是否繁琐.在项目开发中,我们一般都会制订构造函数尽量简单,尽可能不抛异常 ...
- hdu 4665 搜索
思路:直接搜索 #include<iostream> #include<cstdio> #include<algorithm> #include<cstrin ...
- oracle 9i 中行转列
示例如下: SELECT deptno , LTRIM(MAX(SYS_CONNECT_BY_PATH(ename,',')) KEEP (DENSE_RANK LAST ORDER BY curr) ...
- jquery的datepicker汉化
$("#date").datepicker({ dateFormat: "yy-mm-dd", monthNames:["1月", &quo ...