本文来一点比较轻松的话题,来写一个客户端,从时间服务器(daytime server) 上取得时间并打印。
1. daytime 服务器
daytime 服务器我们不用自己写了,这个网站列举了很多现成的 daytime 服务器地址:http://tf.nist.gov/tf-cgi/servers.cgi , 剩下的事情我们就只管写客户端。
daytime 服务器使用的是 TCP 协议,默认端口是 13。
目前测得可用的 daytime 服务器有下面几个(不同时间段可能会不一样):
sttime.carsoncity.k12.mi.us
wwv.nist.gov
time.nist.gov
utcnist.colorado.edu
time-d.nist.gov
其它的服务器在网络状况好的情况下可以访问,差的情况下也总是出现连接超时的情况。具体看网络环境吧。
2. 时间获取客户端
非常简单,客户端连接上 daytime 服务器后,直接从套接字 read 就行了,然后将读取到的数据打印在屏幕上。
本文使用的程序托管在 gitos 上:http://git.oschina.net/ivan_allen/unp
如果你已经 clone 过这个代码了,请使用 git pull
更新一下。本节程序所使用的程序路径是 unp/program/nonblockio/biotimecli
.
代码不长,直接贴出来:
#include "common.h"
char hostname[64];
int port;
void client_routine();
void doClient(int sockfd);
int main(int argc, char* argv[]) {
Args args = parsecmdline(argc, argv);
WARNING("Usage: %s [-h hostname] [-p port]\n", argv[0]);
// 随便选了一个默认的
SETSTR(args, hostname, "h", "nisttime.carsoncity.k12.mi.us");
SETINT(args, port, "p", 13);
client_routine();
}
void client_routine() {
int ret, sockfd;
struct sockaddr_in servaddr;
ret = resolve(hostname, port, &servaddr);
if (ret < 0) ERR_EXIT("resolve");
sockfd = socket(AF_INET, SOCK_STREAM, 0);
if (sockfd < 0) ERR_EXIT("socket");
// 连接服务器
ret = connect(sockfd, (struct sockaddr*)&servaddr, sizeof(servaddr));
if (ret < 0) ERR_EXIT("socket");
doClient(sockfd);
close(sockfd);
}
void doClient(int sockfd) {
char buf[4096];
int nr;
while(1) {
nr = iread(sockfd, buf, 4096);
if (nr < 0) ERR_EXIT("iread");
else if (nr == 0) {
break;
}
else {
writen(STDOUT_FILENO, buf, nr);
}
}
}
3. 实验
3.1 实验一:正常情况
图1 可以正常获取到时间
可以看到,整个程序的运行时间约为 926 ms,毕竟这个服务器(nisttime.carsoncity.k12.mi.us)都在美国 Georgia 州的 Macon 市,所以有点慢。
3.2 实验二:连接超时
图2 连接超时
这次特意选了一个连接超时的,可以看到整个过程持续了约 127 s 左右。
注:早上我连接 wolfnisttime.com 的时候是不超时的,这应该和网络环境有关。如果你实验的时候连接成功也不要奇怪,只能说明你运气太好^_^
4. 总结
编写这个客户端的目的,是为了引出下一个知识点——非阻塞 i/o 上调用 connect. 不同于非阻塞 i/o 读写,在非阻塞 i/o 上调用 connect 要复杂的多。
- 掌握时间获取客户端编写