accept函数由TCP服务器调用,用于从以完成连接队列队头返回下一个以完成连接
如果以完成队列为空,那么进程被投入睡眠(假定套接字为默认的阻塞方式)
int acceptn ( int sockfd, struct sockaddr * cliaddr,socklen_t * addrlen);
参数cliaddr和addrlen用来返回已连接的对端进程的协议地址
addrlen是值-结果参数:
调用前,我们将由*addrlen所引用的整数值置为由cliaddr所指的套接字地址结构的长度,
返回时,该整数值即为由内核存放来自该套接字地址结构内的确定字节数
如果accept成功,那么其返回值是由内核自动生成的一个全新的描述符,代表与所返回客户的TCP连接
在讨论accept函数时,我们称它的地一个参数为监听套接字描述符(由socket创建,随后用作bind和listen的第一参数描述符)
区分这2个套接字非常重要
一个服务器通常仅仅创建一个监听套接字,它在该服务器的生命周期内一直存在
内核为每个由服务器进程接受的客户连接创建一个以连接套接字(也就是说对于它的TCP三次握手过程已经完成)
当服务器完成对其某个给定客户的服务时,相应的以连接套接字就被关闭了
本函数最多返回3个值:
一个既可能是新套接字描述符也可能是出错指示的整数、客户进程的协议地址(由cliaddr指针所指)、该地址的大小(由addrlen指针所指)
如果对于客户协议地址不敢兴趣,可以把cliaddr和addrlen置为空指针
===================================
显示客户IP地址和端口号的时间获取服务器程序
====================================
1.定义2个新的变量:
len:值-结果变量
cliaddr:存放客户的协议地址
2.创建socket
3.填充服务器地址结构
4.将服务器地址和套接字描述符绑定
5.监听连接请求
6.将len初始化为套接字地址结构的大小
7.将指向cliaddr结构的指针和指向len的指针分别作为accept的第二和第三个参数
8.调用inet_ntop将套接字地址结构中32位IP地址转换为一个点分十进制数ASCII字符串
9.调用ntohs将16位的端口号从网络字节序转换为主机字节序
10.调用time获得系统时间
11.调用ctime控制时间格式
12.将时间写入已经连接的套接字中
13.关闭连接
#include "unp.h"
#include <time.h>
int
main(int argc, char **argv)
{
int listenfd, connfd;
socklen_t len;
struct sockaddr_in servaddr, cliaddr;
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(13); /* daytime server */
Bind(listenfd, (SA *) &servaddr, sizeof(servaddr));
Listen(listenfd, LISTENQ);
for ( ; ; ) {
len = sizeof(cliaddr);
connfd = Accept(listenfd, (SA *) &cliaddr, &len);
printf("connection from %s, port %d\n",
Inet_ntop(AF_INET, &cliaddr.sin_addr, buff, sizeof(buff)),
ntohs(cliaddr.sin_port));
ticks = time(NULL);
snprintf(buff, sizeof(buff), "%.24s\r\n", ctime(&ticks));
Write(connfd, buff, strlen(buff));
Close(connfd);
}
}