|
程序功能:
服务器端给客户端提供时间信息服务。
服务器端代码:
#include <sys/types.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <strings.h>
#include <time.h>
#include <string.h>
#define MAXLINE 4096
#define SA struct sockaddr
#define LISTENQ 1024
int main(int argc, char **argv)
{
int listenfd, connfd;
struct sockaddr_in servaddr;
char buff[MAXLINE];
time_t ticks;
listenfd = socket(AF_INET, SOCK_STREAM, 0);
bzero (&servaddr, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
servaddr.sin_port = htons(25589);
bind (listenfd, (SA *)&servaddr, sizeof(servaddr));
listen(listenfd, LISTENQ);
for(; ;) {
connfd = accept (listenfd, (SA *)NULL, NULL);
ticks = time(NULL);
snprintf (buff, sizeof(buff), "%.24s\r\n", ctime(&ticks));
write(connfd, buff, strlen(buff));
close(connfd);
}
} 程序说明:
指定 IP 地址为 INADDR_ANY,这样就允许服务器在任意接口上接受客户连接(假定服务器主机有多个接口)。
在 socket() 函数创建套接口后,,通过 listen() 函数将此套接口变换成一个监听套接口,它使系统内核接受来自客户的连接。
socket,bind 和 listen 是任何 TCP 服务器用于准备所谓的监听描述符(listening descriptor,上面程序中为 listenfd)通常的三个步骤。
常数 LISTENQ 指定系统内核允许在这个监听描述符上排队的最大客户连接数。因为本服务器每次只处理一个客户端的连接,所以如果有多个客户端连接过来,后来,那么后来的客户就要进行排队。
一般情况下,服务器进程在调用 accept() 函数后处于睡眠状态,它等待客户的连接和内核对他的接受。TCP 连接使用三路握手(three-way handshake,也称为三次握手)来建立,当握手完毕时,accept() 函数返回,其返回值是一个称为已连接描述符(connected descriptor)的新描述符。此描述符用于与新客户的通信。
服务器通过调用 close() 关闭与客户的连接。它引发通常的 TCP 连接终止序列;每个方向上发送一个 FIN ,每个 FIN 又由对方确认。
客户端代码:
#include <sys/types.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <strings.h>
#define MAXLINE 4096
#define SA struct sockaddr
int main(int argc, char **argv)
{
int sockfd, n;
char recvline[MAXLINE + 1];
struct sockaddr_in servaddr;
if (argc != 2) {
printf ("usage: daytime <IP address>\n");
exit (1);
}
if ( (sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
perror ("socket error\n");
exit (1);
}
bzero (&servaddr, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_port = htons(25589);
if (inet_pton(AF_INET, argv[1], &servaddr.sin_addr) <= 0) {
printf ("inet_pton error for %s\n", argv[1]);
exit (1);
}
if (connect(sockfd, (SA *)&servaddr, sizeof(servaddr)) < 0) {
perror("connect error");
exit (1);
}
while ( (n = read(sockfd, recvline, MAXLINE)) > 0) {
recvline[n] = '\0';
if (fputs (recvline, stdout) == EOF) {
perror ("fputs error");
exit (1);
}
}
if (n < 0) {
perror ("read error");
exit (1);
}
exit (0);
} 程序说明:
socket() 函数创建网际(AF_INET)字节流(SOCK_STREAM)套接口。
这里,使用了 25589 端口,服务器端也用这个端口。如使用较小的 13 端口,那么会收到连接被拒绝的提示,这是因为任何小于1024的端口只能分配给超级用户进程的使用。
运行与输出:$ ./daytime_client 127.0.0.1
Wed Dec 9 12:02:44 2009 关于程序中使用的函数及相关介绍见:
http://www.groad.net/bbs/read.php?tid-945.html (套接字)
http://www.groad.net/bbs/read.php?tid-952.html (htonl(),htons(),ntohl(),ntons() 函数)
http://www.groad.net/bbs/read.php?tid-1067.html (close 函数)
http://www.groad.net/bbs/read.php?tid-955.html (inet 系列函数) |
|