贵州碧江区住房和城乡建设局网站热搜词排行榜关键词
iperf3是一个非常强大的工具,它是用C语言编写的。同时iperf3也是用C语言实现面向对象编程的典范,他以数据结构+函数指针为基础,非常好的用C语言实现面向对象的编程的三大特征:封装,继承,多态。相信通过阅读iperf3的源代码不仅能帮助我们更好的理解与使用iperf3,也可以帮助我们去更好的理解C++及其它面向对象的语言里的类-对象-实例化-多态-模板等概念的本质。
main函数,代码流程
在src/main.c文件里,我们可以找到main函数,除了用TEST_PROC_AFFINITY宏括起来的部分外(TEST_PROC_AFFINITY编译宏是一个未定义宏,所以这部分代码是未启用的,为了阅读方便,在本文中删除了),我们可以看到整个main函数分成4大步骤:
- 第一步,对象实例化,创建对象test
- 第二步,根据用户在终端命令行界面输入参数,初始化对象test
- 第三步,运行对象test,处理测试过程
- 如果是客户端:发起连接建立,并发送数据包,开始测试
- 如果是服务端:监听端口,等待接收处理客户端发过来的数据包,开始测试过程
- 第四步,测试结束,删除销毁对象test
int
main(int argc, char **argv)
{struct iperf_test *test;#ifdef TEST_PROC_AFFINITY//这部分代码是未实现的功能,删除
#endif//第一步对象实例化,创建对象test,并用iperf_defaults()对test里的变量和指针赋默认值test = iperf_new_test();if (!test)iperf_errexit(NULL, "create new test error - %s", iperf_strerror(i_errno));iperf_defaults(test); /* sets defaults *///第二步,根据用户在终端命令行界面输入参数,初始化对象testif (iperf_parse_arguments(test, argc, argv) < 0) {iperf_err(test, "parameter error - %s", iperf_strerror(i_errno));fprintf(stderr, "\n");usage_long(stdout);exit(1);}//第三步,运行对象test,处理测试过程if (run(test) < 0)iperf_errexit(test, "error - %s", iperf_strerror(i_errno));//第四步,测试结束,删除销毁对象testiperf_free_test(test);return 0;
}
第一步,测试实例的创建–创建对象test
文章开头介绍过,iperf3是用C语言实现面向对象编程的典范, 所以这里第一步,调用iperf_new_test创建对象,C语言里不支持class, 所以这里用struct里:定义数据+函数指针的方式,来实现类的数据属性+成员函数。然后通过malloc给结构指针附值的方式,实现对象(全部结构指针)test的创建与分配内存(class创建对象的本质是给对象分配内存)。以及给test的settings和bitrate_limit_intervals_traffic_bytes二个struct指针进入分配内存。
struct iperf_test *
iperf_new_test()
{struct iperf_test *test;//通过malloc给test指针分配内存---------对象实例化test = (struct iperf_test *) malloc(sizeof(struct iperf_test));if (!test) {i_errno = IENEWTEST;return NULL;}/* initialize everything to zero */memset(test, 0, sizeof(struct iperf_test));//通过malloc给test->settings指针分配内存test->settings = (struct iperf_settings *) malloc(sizeof(struct iperf_settings));if (!test->settings) {free(test);i_errno = IENEWTEST;return NULL;}memset(test->settings, 0, sizeof(struct iperf_settings));//通过malloc给test->bitrate_limit_intervals_traffic_bytes 指针分配内存test->bitrate_limit_intervals_traffic_bytes = (iperf_size_t *) malloc(sizeof(iperf_size_t) * MAX_INTERVAL);if (!test->bitrate_limit_intervals_traffic_bytes) {free(test->settings);free(test);i_errno = IENEWTEST;return NULL;}memset(test->bitrate_limit_intervals_traffic_bytes, 0, sizeof(sizeof(iperf_size_t) * MAX_INTERVAL));/* By default all output goes to stdout */test->outfile = stdout;return test;
}
第二步,测试实例的初始化–根据输入参数,初始化对象test
这一步,本质上是,调用iperf_defaults对test进行初步的初始化,对test里的数据和函数附默认值(比如tcp, udp, sctp分别调用什么函数去进行数据发送与接收,调用什么函数进行连接的函数指针的初始化),这里我们可以把依赖于配置而附不同的函数的函数指针的过程视为虚接口的初始化。这个函数的调用可以视为构造函数的调用。然后读取用户在终端命令行界面上输入的参数,对test进行变量与方法函数的初始化(配置各种属性,以及根据输入参数的不同,比如是测试UDP还是TCP,是客户端还是服务端等等)。
以-c和-s二个参数为例,如果用户输入的命令行为iperf3 -c 则将test初始化为发送端模式,以便后面第三步调用run时可以找到正确的方法(与客户端、服务端,TCP/UDP相应的回调函数)
int
iperf_defaults(struct iperf_test *testp)
{struct protocol *tcp, *udp;
#if defined(HAVE_SCTP_H)struct protocol *sctp;
#endif /* HAVE_SCTP_H */testp->omit = OMIT;testp->duration = DURATION;testp->diskfile_name = (char*) 0;testp->affinity = -1;testp->server_affinity = -1;TAILQ_INIT(&testp->xbind_addrs);
#if defined(HAVE_CPUSET_SETAFFINITY)CPU_ZERO(&testp->cpumask);
#endif /* HAVE_CPUSET_SETAFFINITY */testp->title = NULL;testp->extra_data = NULL;testp->congestion = NULL;testp->congestion_used = NULL;testp->remote_congestion_used = NULL;testp->server_port = PORT;//以代码省略
}int
iperf_parse_arguments(struct iperf_test *test, int argc, char **argv)
{static struct option longopts[] ={
/*
此处省略大量代码
*/{"server", no_argument, NULL, 's'},{"client", required_argument, NULL, 'c'},
/*
此处省略大量代码
*/{"version4", no_argument, NULL, '4'},{"version6", no_argument, NULL, '6'},
/*
此处省略大量代码
*/{NULL, 0, NULL, 0}};while ((flag = getopt_long(argc, argv, "p:f:i:D1VJvsc:ub:t:n:k:l:P:Rw:B:M:N46S:L:ZO:F:A:T:C:dI:hX:", longopts, NULL)) != -1) {switch (flag) {
/*
此处省略大量代码
*/ case 's':if (test->role == 'c') {i_errno = IESERVCLIENT;return -1;}iperf_set_test_role(test, 's');break;case 'c':if (test->role == 's') {i_errno = IESERVCLIENT;return -1;}iperf_set_test_role(test, 'c');}break;case '4':test->settings->domain = AF_INET;break;case '6':test->settings->domain = AF_INET6;break;
/*
此处省略大量代码
*/}
/*
此处省略大量代码
*/if ((test->role != 'c') && (test->role != 's')) {i_errno = IENOROLE;return -1;}
/*
此处省略大量代码
*/return 0;
}
第三步,测试实例的执行–运行对象test,处理测试过程
在这里,程序会根据自己是运行在服务端还是客户端模式,执行不同的测试流程:
- 在服务端:
程序会打开socket,监听并等待连接;连接成功后,就不停的把客户端送过来的数据读取出来并丢弃,直到连接断开;继续监听并等待新的连接。 - 在客户端:
- 程序会打开socket,去连接服务端,连接成功后,按指定测试方式发送数据包,直到测试结束。
static int
run(struct iperf_test *test)
{/*代码删除,内容略*/free(test);
}