一、常用函数
1. socket()
int socket(int family, int type, int protocol);
socket()
打开一个网络通讯端口,如果成功的话,返回一个文件描述符,应用程序可以像读写文件一样用read/write
在网络上收发数据,如果socket()
调用出错则返回-1。对于IPv4,family
参数指定为AF_INET
。对于TCP协议,type
参数指定为SOCK_STREAM
,表示面向流的传输协议。如果是UDP协议,则type
参数指定为SOCK_DGRAM
,表示面向数据报的传输协议。protocol
参数的介绍从略,指定为0即可。
2. bind()
int bind(int sockfd, const struct sockaddr *myaddr, socklen_t addrlen);
服务器程序所监听的网络地址和端口号通常是固定不变的,客户端程序得知服务器程序的地址和端口号后就可以向服务器发起连接,因此服务器需要调用bind绑定一个固定的网络地址和端口号。
bind()
成功返回0,失败返回-1。bind()
的作用是将参数sockfd
和myaddr
绑定在一起,使sockfd
这个用于网络通讯的文件描述符监听myaddr
所描述的地址和端口号。前面讲过,struct sockaddr *
是一个通用指针类型,myaddr
参数实际上可以接受多种协议的sockaddr
结构体,而它们的长度各不相同,所以需要第三个参数addrlen
指定结构体的长度
3. listen()
int listen(int sockfd, int backlog);// backlog取值0~5.
典型的服务器程序可以同时服务于多个客户端,当有客户端发起连接时,服务器调用的accept()
返回并接受这个连接,如果有大量的客户端发起连接而服务器来不及处理,尚未accept
的客户端就处于连接等待状态,listen()
声明sockfd
处于监听状态,并且最多允许有backlog
个客户端处于连接待状态,如果接收到更多的连接请求就忽略。listen()
成功返回0,失败返回-1。
4. accept()
int accept(int sockfd, struct sockaddr *cliaddr, socklen_t *addrlen);
三方握手完成后,服务器调用accept()
接受连接,如果服务器调用accept()
时还没有客户端的连接请求,就阻塞等待直到有客户端连接上来。cliaddr
是一个传出参数,accept()
返回时传出客户端的地址和端口号。addrlen
参数是一个传入传出参数(value-result argument),传入的是调用者提供的缓冲区cliaddr
的长度以避免缓冲区溢出问题,传出的是客户端地址结构体的实际长度(有可能没有占满调用者提供的缓冲区)。如果给cliaddr
参数传NULL,表示不关心客户端的地址。
二、简单实现服务器编程(C语言)
unsigned char TcpRecvBuf[1520]; /* 接收缓冲区 */
static void Task_WebServer (void *pdata)
{
struct sockaddr_in server, client;
int sock, client_socket;
socklen_t len;
server.sin_family = AF_INET; /* 地址类型为IPv4 */
server.sin_port = htons( 80 ); /* 设置服务器的端口:80端口 */
server.sin_addr.s_addr = htonl( INADDR_ANY ); /* 监听本地任意网卡 */
sock = socket( AF_INET, SOCK_STREAM, 0 ); /* 使用TCP连接 */
bind( sock, (struct sockaddr *)&server, sizeof( server ) );/* 绑定本地接口 */
listen( sock, 5 ); /* 进入监听模式 */
while(1) {
len = sizeof(client);
client_socket = accept( sock, (struct sockaddr *)&client, &len );/* 接受一个连接,会阻塞程序 */
if(client_socket != -1) {
/*
* 接收数据,将会阻塞程序
*/
if ((read(client_socket, TcpRecvBuf, sizeof(TcpRecvBuf))) > 0) {
/*
* 简单的检查数据是否符合HTTP协议,数据内容开头是“GET /”
*/
if(TcpRecvBuf[0] == 'G' && TcpRecvBuf[1] == 'E' &&
TcpRecvBuf[2] == 'T' && TcpRecvBuf[3] == ' ' &&
TcpRecvBuf[4] == '/') {
/**
* 下面为GET请求返回的数据,用户可以自己定义
*/
//write(client_socket, http_html_hdr,sizeof(http_html_hdr));
//write(client_socket, indexdata,sizeof(indexdata));
}
}
close( client_socket ); /* 关闭此连接 */
}
/** 这里可以延时一定时间 */
}
}