一、在之前设计的网络编程服务器中,涉及到的服务器都是最为简单的、一对一的服务器,也就是只与一个客户端建立通信。然而在网络程序里面,一般来说都是许多客户对应一个服务器,为了处理客户的请求, 对服务端的程序就提出了特殊的要求。
二、循环服务器和并发服务器
1、循环服务器:循环服务器描述了在一个时刻只处理一个请求的服务器实现方式,通过在单线程内设置循环控制实现对多个客户端请求的逐一响应,这种服务器的设计、编程、调试和修改往往比较容易去实现。在循环执行的服务器对预期的负载能提供足够的反应速度时常使用这种类型的服务器。循环服务器有有UDP循环服务器和TCP循环服务器两种类型。
①UDP循环服务器:
UDP循环服务器的实现方法:UDP服务器每次从套接字上读取一个客户端的请求,再对请求进行处理,然后将结果返回给客户机。其具体模型如下所示:
socket(...);
bind(...);
while(1)
{
recvfrom(...);
process(...);
sendto(...);
}
对于此类服务器,由于UDP是非面向连接的,没有一个客户端可以老是占住服务端,因此UDP循环服务器对于每一个客户机的请求总是能够满足。
下面是UDP循环服务器的运用:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#define PORT 3212
#define MAX_SIZE 512
int main()
{
int sockfd;
int len = sizeof(struct sockaddr);
char buf[MAX_SIZE];
char buffer[MAX_SIZE];
struct sockaddr_in serv_addr;
//创建套接字,IPV4/UDP
sockfd = socket(AF_INET,SOCK_DGRAM,0);
if(sockfd < 0)
{
printf("create socket error!\n");
exit(1);
}
//填充服务器信息
bzero(&serv_addr,sizeof(struct sockaddr_in));
serv_addr.sin_family = AF_INET;
serv_addr.sin_port = htons(PORT);
serv_addr.sin_addr.s_addr = inet_addr("192.168.1.132");
//绑定套接字
if(bind(sockfd,(struct sockaddr*)&serv_addr,sizeof(struct sockaddr)) < 0)
{
printf("bind error!\n");
exit(1);
}
//循环接收网络上发送来的数据并回复消息
while(1)
{
//接收数据
if(recvfrom(sockfd,buf,MAX_SIZE,0,(struct sockaddr*)&serv_addr,&len) < 0)
{
printf("recv error!\n");
exit(1);
}
printf("recv is: %s\n ",buf);
printf("write some text:");
scanf("%s",buffer);
//发送数据
if(sendto(sockfd,buffer,MAX_SIZE,0,(struct sockaddr*)&serv_addr,len) < 0)
{
printf("send error!\n");
fprintf(stderr,"send error:%s\n",strerror(errno));
exit(1);
}
}
//关闭连接
close(sockfd);
return 0;
}
②TCP循环服务器:
TCP服务器接受一个客户端的连接,然后处理,完成了这个客户的所有请求后,断开连接。
socket(...);
bind(...);
listen(...);
while(1)
{
accept(...);
process(...);
close(...);
}
因为TCP是面向连接的,故TCP循环服务器在同一时刻一次只能处理一个客户端的请求。只有在这个客户的所有请求都满足后, 服务器才可以继续后面的请求。这样如果有一个客户端占住服务器不放时,其它的客户机都不能工作了,因此,TCP服务器一般很少用循环服务器模型。
TCP循环服务器的运用:
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <netdb.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#define portnumber 3000 //通信端口号
int main(int argc, char *argv[])
{
int sockfd,new_fd;
struct sockaddr_in server_addr;
struct sockaddr_in client_addr;
int sin_size;
int nbytes;
char buffer[1024];
/* 服务器端开始建立sockfd描述符 */
if((sockfd=socket(AF_INET,SOCK_STREAM,0))==-1) // IPV4协议;TCP协议
{
fprintf(stderr,"Socket error:%s\n\a",strerror(errno));
exit(1);
}
/* 服务器端填充 sockaddr结构 */
bzero(&server_addr,sizeof(struct sockaddr_in)); // 初始化,置0
server_addr.sin_family=AF_INET; // IPV4
server_addr.sin_addr.s_addr=htonl(INADDR_ANY); // (将本机器上的long数据转化为网络上的long数据)和任何主机通信,INADDR_ANY :可以接收任意IP地址
//server_addr.sin_addr.s_addr=inet_addr("192.168.1.132"); //用于绑定到一个固定IP,inet_addr用于把数字加格式的ip转化为整形ip
server_addr.sin_port=htons(portnumber); // (将本机器上的short数据转化为网络上的short数据)端口号
/* 捆绑sockfd描述符到IP地址 */
if(bind(sockfd,(struct sockaddr *)(&server_addr),sizeof(struct sockaddr))==-1)
{
fprintf(stderr,"Bind error:%s\n\a",strerror(errno));
exit(1);
}
/* 设置允许连接的最大客户端数 */
if(listen(sockfd,5)==-1)
{
fprintf(stderr,"Listen error:%s\n\a",strerror(errno));
exit(1);
}
while(1)
{
/* 服务器阻塞,直到客户程序建立连接 */
sin_size=sizeof(struct sockaddr_in);
if((new_fd=accept(sockfd,(struct sockaddr *)(&client_addr),&sin_size))==-1)
{
fprintf(stderr,"Accept error:%s\n\a",strerror(errno));
exit(1);
}
fprintf(stderr,"Server get connection from %s\n",inet_ntoa(client_addr.sin_addr)); // 将网络地址转换成.字符串
if((nbytes=read(new_fd,buffer,1024))==-1)
{
fprintf(stderr,"Read Error:%s\n",strerror(errno));
exit(1);
}
buffer[nbytes]='\0';
printf("Server received %s\n",buffer);
/* 这个通讯已经结束 */
close(new_fd);
/* 循环下一个 */
}
/* 结束通讯 */
close(sockfd);
exit(0);
}
这种服务器有下面这些缺点:
*若服务器正在处理一个客户端请求时另一个请求到来,系统会将该新的请求排队,第二个请求须等待第一个请求处理完才能开始,这就造成在客户请求过于频繁也就是客户请求过多时服务器的请求队伍会越来越长,响应时间也就越来越久。
*如果一个服务器的设计的处理能力为处理K个客户端,而每个客户端每秒发送N个请求,则此服务器的请求处理时间必须小于每个请求1\(K*N)s。若服务器无法达到该要求则会造成客户的请求队列溢出。这也就衍生出了并发服务器。
2、并发服务器:
并发服务器在同一个时刻可以响应多个客户端的请求,将并发引入的原因是需要给多个客户提供快速的响应时间。并发方式处理多个客户的请求是目前最为广泛运用的服务器类型,常见的并发服务器的实现是多线程机制(也可以使用多进程,不过这种方法消耗资源太大,通常不使用),在多线程机制中服务器主线程为每个到来的客户请求创建一个子线程进行服务,这类服务器的编写规则通常如下:
第一部分的代码负责监听并接收客户请求,为客户请求创建新的服务器线程;第二部分的代码负责处理单个客户的请求。关于多线程机制并发服务器的设计流程如下:
socket(...);
bind(...);
listen(...);
while(1)
{
accept(...);
if(fork(..)==0) //创建线程处理请求
{
process(...);
close(...);
exit(...); //处理结束,退出线程
}
close(...);
}
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <netdb.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#define portnumber 3000
void * read_msg(void *arg) //处理客户请求(读取客户端数据)
{
int fd = *((int *)arg);
int nread = 0;
char buffer[1024];
while((nread = read(fd,buffer,sizeof(buffer))) > 0)
{
buffer[nread] = '\0';
printf("get client message: %s\n",buffer);
memset(buffer,0,sizeof(buffer));
}
}
int main(int argc, char *argv[])
{
int sockfd,new_fd;
struct sockaddr_in server_addr;
struct sockaddr_in client_addr;
int sin_size;
int nbytes;
char buffer[1024];
pthread_t id;
/* 服务器端开始建立sockfd描述符 */
if((sockfd=socket(AF_INET,SOCK_STREAM,0))==-1) // AF_INET:IPV4;SOCK_STREAM:TCP
{
fprintf(stderr,"Socket error:%s\n\a",strerror(errno));
exit(1);
}
/* 服务器端填充 sockaddr结构 */
bzero(&server_addr,sizeof(struct sockaddr_in)); // 初始化,置0
server_addr.sin_family=AF_INET; // Internet
server_addr.sin_addr.s_addr=htonl(INADDR_ANY); // 和任何主机通信,INADDR_ANY可以接收任意IP地址
//server_addr.sin_addr.s_addr=inet_addr("192.168.1.132"); //用于绑定到一个固定IP,inet_addr用于把数字加格式的ip转化为整形ip
server_addr.sin_port=htons(portnumber); // (将本机器上的short数据转化为网络上的short数据)端口号
/* 捆绑sockfd描述符到IP地址 */
if(bind(sockfd,(struct sockaddr *)(&server_addr),sizeof(struct sockaddr))==-1)
{
fprintf(stderr,"Bind error:%s\n\a",strerror(errno));
exit(1);
}
/* 设置允许连接的最大客户端数 */
if(listen(sockfd,5)==-1)
{
fprintf(stderr,"Listen error:%s\n\a",strerror(errno));
exit(1);
}
while(1)
{
/* 服务器阻塞,直到客户程序建立连接 */
sin_size=sizeof(struct sockaddr_in);
printf("accepting!\n");
if((new_fd=accept(sockfd,(struct sockaddr *)(&client_addr),&sin_size))==-1)
{
fprintf(stderr,"Accept error:%s\n\a",strerror(errno));
exit(1);
}
fprintf(stderr,"Server get connection from %s\n",inet_ntoa(client_addr.sin_addr)); // 将网络地址转换成.字符串
pthread_create(&id,NULL,(void *)read_msg,(void *)&new_fd); //创建线程
}
/* 结束通讯 */
close(sockfd);
exit(0);
}
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <netdb.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#define MY_PORT 3000
int main(int argc ,char **argv)
{
int listen_fd,accept_fd;
struct sockaddr_in client_addr;
int n;
int nbytes;
if((listen_fd=socket(AF_INET,SOCK_STREAM,0))<0)
{
printf("Socket Error:%s\n\a",strerror(errno));
exit(1);
}
bzero(&client_addr,sizeof(struct sockaddr_in));
client_addr.sin_family=AF_INET;
client_addr.sin_port=htons(MY_PORT);
client_addr.sin_addr.s_addr=htonl(INADDR_ANY);
n=1;
/* 如果服务器终止后,服务器可以第二次快速启动而不用等待一段时间 */
setsockopt(listen_fd,SOL_SOCKET,SO_REUSEADDR,&n,sizeof(int));
if(bind(listen_fd,(struct sockaddr *)&client_addr,sizeof(client_addr))<0)
{
printf("Bind Error:%s\n\a",strerror(errno));
exit(1);
}
listen(listen_fd,5);
while(1)
{
accept_fd=accept(listen_fd,NULL,NULL);
if((accept_fd<0)&&(errno==EINTR))
continue;
else if(accept_fd<0)
{
printf("Accept Error:%s\n\a",strerror(errno));
continue;
}
if((n=fork())==0)
{
/* 子进程处理客户端的连接 */
char buffer[1024];
if((nbytes=read(accept_fd,buffer,1024))==-1)
{
fprintf(stderr,"Read Error:%s\n",strerror(errno));
exit(1);
}
buffer[nbytes]='\0';
printf("Server received %s\n",buffer);
close(listen_fd);
close(accept_fd);
exit(0);
}
else
close(accept_fd);
}
}