1:客户服务程序
如何写好一个客户服务程序也很简单,现在先从最基本的来讲述,写一个读取时间的客户程序,以后再对其进行进一步优化。
具体流程是:
a)创建套接字,sockfd用于标示
b)先清空套接字地址结构,再对sin_family/sin_port/sin_addr三者赋值(其实sockaddr_in结构中还有其他一些变量,只是一般用不到,清空时也对其他清空)
c)结构体定义完毕,就调用connet()函数将套接字sockfd和结构中的ip地址指向的服务器进行连接
#include "unp.h"
int main(int argc,char **argv)
{
int sockfd,n;
struct sockaddr_in servaddr;
char recvline[MAXLINE + 1];
if(2 != argc){
fprintf(stderr,"usage: a.out <ipaddress>");
exit(1);
}
//1:创建套接字,sockfd用于标示
sockfd = socket(AF_INET,SOCK_STREAM,0);
if(sockfd < 0){
fprintf(stderr,"socket error");
exit(1);
}
//2:先清空套接字地址结构,再对sin_family/sin_port/sin_addr三者赋值
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){
fprintf(stderr,"inet_pron error!");
}
//3:将套接字sockfd和结构中的ip地址指向的服务器进行连接
if( connect(sockfd,(struct sockaddr *)&servaddr,sizeof(servaddr)) < 0){
fprintf(stderr,"connect error");
exit(1);
}
//读取发送回来的信息
while( (n = read(sockfd,recvline,MAXLINE)) > 0){
recvline[n] = '\0';
if(fputs(recvline,stdout) == EOF){
fprintf(stderr,"fputs error");
exit(1);
}
}
//判断读取是否正常终止
if(n < 0){
fprintf(stderr,"read error");
exit(1);
}
exit(0);
}
注意:
1、sockfd = socket(AF_INET,SOCK_STREAM,0);三个参数相对固定,第一个为协议族(AF_INET表示IPV4);第二个是字节流SOCK_STREAM
2、htons(13) 是将主机字节序转换成网络字节序,这样才能在网络上传输,到达对方主机是再调用ntohs(),就能得到正确的端口号
3、inet_pton(AF_INET,argv[1],&servaddr.sin_addr) 将argv[1]的ip地址转换并付给servaddr.sin_addr结构体,注意这里是结构体,其有一个参数
4、connect(sockfd,(struct sockaddr *)&servaddr,sizeof(servaddr)) 连接的时候,我们要将定义的结构servaddr转化为通用结构struct sockaddr *,类似c中的void *
2:服务器程序
具体流程:
a)创建套接字,现在是普通的套接字
b)结构体清空,并对响应的变量初始化
c)将套接字和结构体绑定,类似客户程序中的connect()函数
d)将套接字转换为监听套接字,这时当客户程序来连接时,可以在该监听套接字上由内核接受
#include "unp.h"
#include <time.h>
int main(int argc,char **argv)
{
int listenfd,connfd;
struct sockaddr_in servaddr;
char buff[MAXLINE]; //存储可读的时间格式
time_t ticks; //用于获取秒
//1:创建套接字,现在是普通的套接字
listenfd = Socket(AF_INET,SOCK_STREAM,0);
//2:结构体清空,并对响应的变量初始化
bzero(&servaddr,sizeof(servaddr));
servaddr.sin_family = AF_INET; //IPV4
servaddr.sin_port = htons(13);
servaddr.sin_addr.s_addr = htonl(INADDR_ANY); //这里用到的是变量,客户程序中用servaddr.sin_addr结构体
//3:将套接字和结构体绑定,类似客户程序中的connect()函数
Bind(listenfd,(SA *)&servaddr,sizeof(servaddr));
//4:将套接字转换为监听套接字,这时当客户程序来连接时,可以在该监听套接字上由内核接受
Listen(listenfd,LISTENQ); //LISTENQ指定最大的连接数
for(;;){
connfd = Accept(listenfd,(SA *)NULL,NULL); //等待接收,无连接时睡眠
ticks = time(NULL); //调用系统函数,获取时间(秒)
snprintf(buff,sizeof(buff),"%.24s\r\n",ctime(&ticks)); //ctime()将秒转化为可读字符串
Write(connfd,buff,strlen(buff));
Close(connfd); //关闭套接字
}
}
注意:
1、servaddr.sin_addr.s_addr = htonl(INADDR_ANY); 这里等号左边是变量s_addr,跟在客户进程中不一样。
2、Accept(listenfd,(SA *)NULL,NULL) 其实也类似connect和bind两个函数的调用,不过最后一个参数为指针,不为整数!是为了“值-结果”来设计的
3:编译及使用
本文程序是在《UNIX网络编程》搭建的平台上实现的,unp.h头文件也是该书中提供的,但编译的时候要加上-lunp参数才行。
编译:gcc daytimecli.c -o daytimecli -lunp
调用:./daytimecli 127.0.0.1 (利用环回地址)