客户端程序
/**************************************
编写一个简单的时候获取客户端的步骤是:
定义网际套接字地址结构
使用socket函数创建网际字符流套接字
设置地址簇结构并绑定端口号
使用inet_pton将点分十进制地址转换成为正确的IP地址到地址簇结构成员中
使用connect函数获取连接
接下来就可以读写网际套接字
**************************************/
/*************************************
复习一些函数:
#include<sys/types.h>
int socket(int domain,int type,int protocol);
该函数创建会话端点,并且返回socket描述符,domain决定了通信的协议,可以取值AF_UNIX(本地通信),AF_INET(IPv4网际通信),AF_INET6(IPv6网际通信)等等;type决定通信的语义,可取的值为SOCK_STREAM(连续可靠双向字节流)等等值。
int inet_pton(int af,const char*src,void *dst);
af确定地址簇,只可以取AF_INET,AF_INET6中的一个;src根据af的取值而不同,对于AF_INET来说,就是点分十进制表示的IP地址,dest代表网络地址结构(在网际地址结构中的成员)。该函数在成功执行的时候返回1,如果src是无效的IP地址,那么返回0,失败返回-1。
int connect(int sockfd,const struct sockaddr *addr,socklent_t addrlen);
该函数建立一个到指定IP的主机,该套接字用sockfd表示。注意第二个参数是要一个通用地址结构指针。
***************************************/
typedef struct sockaddr* SA;
int main(int argc,char**argv){
int sockfd,n;
struct sockaddr_in addr;
if(argc!=2){
printf("usage: a.out ip_adrress\n");
exit(0);
}
if((sockfd=socket(AF_INET,SOCK_STREAM,0))<0){
printf("create AF_INET socket error\n");
exit(0);
}
//set addr
bzero(&addr,sizeof(addr));
addr.sin_family=AF_INET;
addr.sin_port=htons(13);
if(inet_pton(AF_INET,argv[1],&addr.sin_addr)<0){
printf("convert dot decimal ip error\n");
exit(0);
}
if(connect(sockfd,(struct sockaddr*)&addr,sizeof(addr))<0){
perror("错误是:\n");
printf("connect to server error\n");
exit(0);
}
const int maxline=4096;
char buff[]="hello,my name is xcl,i want to connect you!";
char buf[maxline+1];
//在服务器不读的情况下,经过验证,会影响下面的while循环
if(write(sockfd,buff,strlen(buff))<0){
printf("sorry,say hello to server error/n");
exit(0);
}
while((n=read(sockfd,buf,maxline))>0){
buf[n]=0;
if(fputs(buf,stdout)<0){
printf("print message from server error/n");
exit(0);
}
printf("current n=%d\n",n);
}
printf("out while current n=%d\n",n);
if(n<0){
perror("错误是:");
printf("error read\n");
}
exit(0);
}
服务器程序
/***********************************
创建一个简单的时间服务器的步骤:
定义地址簇结构
使用socket创建普通套接字
调用bind函数将刚刚创建的套接字绑定到地址结构
调用listen函数将普通套接字转换成监听套接字
调用accept函数在指定的监听套接字进行监听
对该套接字进行读写
***********************************/
/***********************************
复习一些函数:
int bind(int sockfd,const struct sockaddr *addr,socklent_t addrlen);
该函数的意图是将一个地址结构与socket描述符进行绑定,因为服务器段的socket创建的时候是没有地址结构与其绑定的。
int listen(int sockfd,int backlog);
该函数的作用是将一个套接字转换成accept函数需要使用的监听套接字。backlog参数定义了等待与服务器进程链接的队列的大小。
int accept(int sockfd,struct sockaddr *addr,socklen_t *addrlen);
该函数的作用是阻塞服务器进程,知道有客户端链接的时候唤醒,addr字段可以返回客户端的地址,accept函数的返回值是建立通信链接的套接字地址。
*************************************/
typedef struct sockaddr* SA;
int main(int argc,char**argv){
int listenfd,connfd;
const int maxqueue=1024;
const int maxline=4096;
char buf[maxline];
char buff[maxline];
struct sockaddr_in addr;
time_t ticks;
if((listenfd=socket(AF_INET,SOCK_STREAM,0))<0){
printf("服务器建立网际流套接字失败\n");
exit(0);
}
bzero(&addr,sizeof(addr));
addr.sin_family=AF_INET;
addr.sin_port=htons(13);
addr.sin_addr.s_addr=htonl(INADDR_ANY);
printf("before bind\n");
if(bind(listenfd,(struct sockaddr*)&addr,sizeof(addr))<0){
printf("bind绑定addr到listenfd失败\n");
perror("错误是:");
exit(0);
}
printf("after bind\n");
if(listen(listenfd,maxqueue)<0){
printf("将listenfd标记成为监听套接字失败\n");
exit(0);
}
int n;
for(;;){
printf("before accept\n");
if((connfd=accept(listenfd,(struct sockaddr*)NULL,NULL))<0){
printf("等待客户链接失败\n");
exit(0);
}
printf("after accept\n");
if((n=read(connfd,buf,maxline))<0){
printf("读取客户端信息失败\n");
exit(0);
}
if(n<0){
printf("error get message from customer\n");
exit(0);
}else{
buf[n]=0;
fputs(buf,stdout);
printf("\n");
}
printf("befor write\n");
ticks=time(NULL);
snprintf(buf,sizeof(buf),"%.24s\r\n",ctime(&ticks));
printf("服务器要返回的时间是:%s\n",buf);
if(write(connfd,buf,strlen(buf))<0){
printf("将时间交给客户失败\n");
exit(0);
}
close(connfd);
}
}