时间获取程序客户端 TCP 协议相关性

本文为senlie原创,转载请保留此地址:http://blog.csdn.net/zhengsenlie

最初代码:

这是一个简单的时间获取客户程序。客户与服务器建立一个TCP连接后,服务器以直观

可读格式简单地送回当前时间和日期

这个例子是协议相关,使用 inet_pton

TCP/IPv4, IPv6 协议相关

IPv4 --> IPv6 (把代码中出现的左边的字符串换为右边的,就变成了IPv6版本的)

sockaddr_in --> sockaddr_in6

AF_INET --> AF_INET6

sin_family --> sin6_family

sin_port --> sin6_port

sin_addr --> sin6_addr

#include    "unp.h"

int
main(int argc, char **argv)
{
int sockfd, n;
char recvline[MAXLINE + 1];
struct sockaddr_inservaddr;

if (argc != 2)
err_quit("usage: a.out <IPaddress>");

//1.创建 TCP 套接字
// socket 函数创建一个网际(AF_INET)字节流(SOCK_STREAM)套接字,即 TCP 套接字。
//该函数返回一个整数描述符,以后的所有函数调用都用它来标识这个套接字
if ( (sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
err_sys("socket error");

    //2.指定服务器的IP地址和端口
    //使用 bzero 把套接字地址结构 servaddr 清零
    //设置地址族为 AF_INET
    //设置端口号为 13,htons 将主机字节序转换为网络字节序的短整形。网络字节序采用的是大端存储
    //inet_pton 将IP地址的点分十进制(eg. 206.168.112.96)转为网络字节序的二进制形式
bzero(&servaddr, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_port   = htons(13);
if (inet_pton(AF_INET, argv[1], &servaddr.sin_addr) <= 0)
err_quit("inet_pton error for %s", argv[1]);

    //3.建立与服务器的连接
    //将 sockfd 套接字与 servaddr 套接字地址结构指定的服务器建立一个 TCP 连接
if (connect(sockfd, (SA *) &servaddr, sizeof(servaddr)) < 0)
err_sys("connect error");

    //4.读入并输出服务器的应答
    //使用 read 从 sockfd 套接字读取服务器的应答,并存储在 recvline 缓冲里
    //使用 fputs 将应答输出到标准 I/O
while ( (n = read(sockfd, recvline, MAXLINE)) > 0) {
recvline[n] = 0;/* null terminate */
if (fputs(recvline, stdout) == EOF)
err_sys("fputs error");
}
if (n < 0)
err_sys("read error");

    //5.终止程序
    //进程终止时会关闭该进程所有打开的描述符,所以 sockfd 套接字会被关闭
exit(0);
}

问题1:必须以点分十进制格式给出服务器的 IP 地址

改善:使用 gethostbyname 和 getservbyname ,并改用2个命令行参数:主机名和服务名

/*
 *TCP/IPv4 协议相关,调用gethostbyname和getservbyname
 * */
#include "unp.h"

int
main(int argc, char **argv)
{
int sockfd, n;
char recvline[MAXLINE + 1];
//struct sockaddr_in --> IPv4 套接字地址结构
struct sockaddr_inservaddr;
    //struct in_addr --> 32位 IPv4 地址,网络字节序
struct in_addr**pptr;
struct in_addr*inetaddrp[2];
struct in_addrinetaddr;

struct hostent*hp;
struct servent*sp;

if (argc != 3)
err_quit("usage: daytimetcpcli1 <hostname> <service>");

    //1.调用 gethostbyname 和 getservbyname
    // gethostbyname 用主机名向 DNS 查询 IP地址
    // 如果查询失败,尝试使用 inet_aton 函数确定其参数是否已是 ASCII 格式的地址
    // 若是则构造一个由相应的地址构成的单元素列表(两个元素,第二个元素设为 NULL)
if ( (hp = gethostbyname(argv[1])) == NULL) {
if (inet_aton(argv[1], &inetaddr) == 0) {
err_quit("hostname error for %s: %s", argv[1], hstrerror(h_errno));
} else {
inetaddrp[0] = &inetaddr;
inetaddrp[1] = NULL;
pptr = inetaddrp;
}
} else {
pptr = (struct in_addr **) hp->h_addr_list;
}
    // getservbyname 用服务名在 /etc/services 里查询服务端口号
if ( (sp = getservbyname(argv[2], "tcp")) == NULL)
err_quit("getservbyname error for %s", argv[2]);

//2.尝试每个服务器主机地址
for ( ; *pptr != NULL; pptr++) {
//创建 TCP 套接字
sockfd = Socket(AF_INET, SOCK_STREAM, 0);

//指定服务器的IP地址和端口
bzero(&servaddr, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_port = sp->s_port;
memcpy(&servaddr.sin_addr, *pptr, sizeof(struct in_addr));
printf("trying %s\n",
  Sock_ntop((SA *) &servaddr, sizeof(servaddr)));

//调用 connect ,建立与服务器的连接
if (connect(sockfd, (SA *) &servaddr, sizeof(servaddr)) == 0)
break; /* 成功 */
err_ret("connect error");

//关闭套接字连接,尝试下一个主机地址
close(sockfd);
}

//3.检查是否失败
if (*pptr == NULL)
err_quit("unable to connect");

//4.读取服务器的应答
while ( (n = Read(sockfd, recvline, MAXLINE)) > 0) {
recvline[n] = 0;/* 字符串的终止符'0' */
Fputs(recvline, stdout);
}
exit(0);
}

问题2:上面的代码是协议相关

改善:使用 getaddrinfo 和 tcp_connect 来同时支持 IPv4 和 IPv6

/**
* TCP,协议无关,调用 getaddrinfo 和 tcp_connect
**/
#include "unp.h"

int
tcp_connect(const char *host, const char *serv)
{
int sockfd, n;
struct addrinfohints, *res, *ressave;

//1.调用 getaddrinfo
//协议地址族为 AF_UNSPEC , 套接字类型为 SOCK_STREAM
bzero(&hints, sizeof(struct addrinfo));
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_STREAM;

if ( (n = getaddrinfo(host, serv, &hints, &res)) != 0)
err_quit("tcp_connect error for %s, %s: %s",
host, serv, gai_strerror(n));
ressave = res;

//2.尝试每个 addrinfo 结构直到成功或到达链表尾
do {
//创建套接字
sockfd = socket(res->ai_family, res->ai_socktype, res->ai_protocol);
if (sockfd < 0)
continue; /* 失败,忽略 */
//建立与服务器的连接
if (connect(sockfd, res->ai_addr, res->ai_addrlen) == 0)
break; /* connect 成功,跳出循环 */

Close(sockfd);/* connect 不成功,关闭套接字 */
} while ( (res = res->ai_next) != NULL);

//3.如果 getaddrinfo 失败或 connect 调用没有一次成功,本函数将终止
if (res == NULL)/* errno set from final connect() */
err_sys("tcp_connect error for %s, %s", host, serv);

//4.调用 freeaddrinfo 清理由 getaddrinfo 返回的动态存储空间
freeaddrinfo(ressave);

//5.返回成功与服务器建立连接的套接字
return(sockfd);
}
/* end tcp_connect */

int
Tcp_connect(const char *host, const char *serv)
{
return(tcp_connect(host, serv));
}

/**
* TCP,协议无关,调用 getaddrinfo 和 tcp_connect
**/
#include "unp.h"

int
main(int argc, char **argv)
{
int sockfd, n;
char recvline[MAXLINE + 1];
socklen_t len;
struct sockaddr_storagess;

if (argc != 3)
err_quit("usage: daytimetcpcli <hostname/IPaddress> <service/port#>");

//1.利用 Tcp_connect 连接到服务器
sockfd = Tcp_connect(argv[1], argv[2]);

//2.显示服务器地址
len = sizeof(ss);
Getpeername(sockfd, (SA *)&ss, &len);
printf("connected to %s\n", Sock_ntop_host((SA *)&ss, len));

//3.读取服务器的应答,并显示到标准输出
while ( (n = Read(sockfd, recvline, MAXLINE)) > 0) {
recvline[n] = 0;/* null terminate */
Fputs(recvline, stdout);
}
exit(0);
}

时间: 2024-10-29 19:05:44

时间获取程序客户端 TCP 协议相关性的相关文章

时间获取程序服务器 TCP 协议相关性

本文为senlie原创,转载请保留此地址:http://blog.csdn.net/zhengsenlie 最初代码: 这是一个简单的时间获取服务器程序.它和时间获取程序客户端一道工作. TCP/IPv4, IPv6 协议相关 IPv4 --> IPv6 (把代码中出现的左边的字符串换为右边的,就变成了IPv6版本的) sockaddr_in --> sockaddr_in6 AF_INET --> AF_INET6 sin_family --> sin6_family sin_p

UNIX网络编程卷1 时间获取程序server TCP 协议相关性

本文为senlie原创.转载请保留此地址:http://blog.csdn.net/zhengsenlie 最初代码: 这是一个简单的时间获取server程序.它和时间获取程序client一道工作. 它是 协议相关,把代码中出现的左边的字符串换为右边的,就变成了IPv6版本号的 IPv4 --> IPv6 sockaddr_in --> sockaddr_in6 AF_INET --> AF_INET6 sin_family --> sin6_family sin_port --&

时间获取程序客户端 TCP 使用非阻塞connect

本文为senlie原创,转载请保留此地址:http://blog.csdn.net/zhengsenlie /** * TCP,使用非阻塞 connect **/ #include "unp.h" int connect_nonb(int sockfd, const SA *saptr, socklen_t salen, int nsec) { int flags, n, error; socklen_t len; fd_set rset, wset; struct timeval t

时间获取程序客户端 UDP 协议无关

本文为senlie原创,转载请保留此地址:http://blog.csdn.net/zhengsenlie /** * UDP,协议无关,调用 getaddrinfo 和 udp_client **/ #include "unp.h" int udp_client(const char *host, const char *serv, SA **saptr, socklen_t *lenp) { int sockfd, n; struct addrinfo hints, *res, *

UNIX网络编程卷1 时间获取程序client TCP 使用非堵塞connect

本文为senlie原创,转载请保留此地址:http://blog.csdn.net/zhengsenlie 1.当在一个非堵塞的 TCP 套接字(可使用 fcntl 把套接字变成非堵塞的)上调用 connect时. connect 将马上返回一个 EINPROGRESS 错误,只是已经发起的 TCP 三路握手继续进行. 接着使用 select 检測 这个连接是否建立成功 2.用途: 1)在TCP三路握手的时候进行其它处理 2)同一时候建立多个连接 3)使用 select 设置超时 /** * T

时间获取程序服务器 UDP 协议无关

本文为senlie原创,转载请保留此地址:http://blog.csdn.net/zhengsenlie /** * UDP 协议无关 调用 getaddrinfo 和 udp_server **/ #include "unp.h" #include <time.h> int main(int argc, char **argv) { int sockfd; ssize_t n; char buff[MAXLINE]; time_t ticks; socklen_t le

获取系统时间&amp;&amp;获取程序某一段执行时间

获取系统时间: CTime  _time; CString m_SystemTime("");  _time=CTime::GetCurrentTime();  m_SystemTime=_time.Format("%Y-%m-%d  %H:%M:%S"); 获取程序某一段执行时间: DWORD  betime(0),endtime(0),sumtime(0); int i=0; betime=GetTickCount(); while(i<10) {  Sl

微信小程序 客户端时间 与 服务端时间

服务端时间 db.serverDate(); 在操作数据库,上传数据的时候可以使用服务端时间 wx.cloud.init();//初始化云 const db = wx.cloud.database(); db.collection('todos').add({ // data 字段表示需新增的 JSON 数据 data: { // _id: 'todo-identifiant-aleatoire', // 可选自定义 _id,在此处场景下用数据库自动分配的就可以了 date: db.server

利用TCP协议实现通信

Socket的TCP通讯 一. socket的通讯原理 服务器端的步骤如下. (1)建立服务器端的Socket,开始侦听整个网络中的连接请求. (2)当检测到来自客户端的连接请求时,向客户端发送收到连接请求的信息,并建立与客户端之间的连接. (3)当完成通信后,服务器关闭与客户端的Socket连接. 客户端的步骤如下. (1)建立客户端的Socket,确定要连接的服务器的主机名和端口. (2)发送连接请求到服务器,并等待服务器的回馈信息. (3)连接成功后,与服务器进行数据的交互. (4)数据处