Linux网络编程服务器模型选择之循环服务器
在网络程序里面,通常都是一个服务器处理多个客户机,为了出个多个客户机的请求,服务器端的程序有不同的处理方式。本节开始介绍Linux下套接字编程的服务器模型选择,主要包括循环服务器模型、并发服务器模型、IO复用服务器模型等,这也是我们常见的几种网络服务器模型。其中基本可以分为两大类,
1. 循环服务器:循环服务器在同一时刻只能响应一个客户端的请求,是比较简单的一种模型;
2. 并发服务器:并发服务器在同一时刻可以响应多个客户端的请求,这里面又有很多分类,接下来会逐步介绍;
循环服务器模型
循环服务器是指对于客户端的请求和连接,服务器逐个进行处理,处理完一个连接后再处理下一个连接,属于串行处理方式,结构比较简单。该模型的算法过程如下:
/* UDP循环服务器模型 */ socket(); bind(); while(true)
{
recvfrom(); process(); sendto();
} close();
/* TCP循环服务器模型 */ socket(); bind(); listen(); while(true)
{ accept(); while(true)
{
recv(); process(); send();
} close();
}
从上面的的流程可以看出,TCP循环服务器比UDP循环服务器多了一个accept的过程,这也是TCP和UDP套接字编程的主要区别。TCP服务器在accept出等待客户端的到来,因为accept函数是阻塞的,因此TCP服务器会在此等待(对accept函数的不同处理是区分各类服务器的一个重要参考依据)。相应地,UDP会在recvfrom阻塞,并等待客户端的连接。
一个循环服务器的例子
下面给出一个简单的循环服务器样子,模拟服务器对外提供时间服务器,等待客户端到来,并返回给客户端服务器的当前时间。
UDP循环服务器
/** UDP循环服务器--server端程序**/
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <time.h>
#include <string.h>
#include <stdio.h>
#define BUFFLEN 1024
#define SERVER_PORT 12345
int main(int argc, char *argv[])
{
int s; //服务器套接字文件描述符
struct sockaddr_in local, to; //本地地址
time_t now; //时间
char buff[BUFFLEN];//收发数据缓冲区
int n = ;
int len = sizeof(to); //建立UDP套接字
s = socket(AF_INET, SOCK_DGRAM, ); //初始化地址
memset(&local, , sizeof(local));
local.sin_family = AF_INET;//AF_INET协议族
local.sin_addr.s_addr = htonl(INADDR_ANY);//任意本地地址
local.sin_port = htons(SERVER_PORT);//服务器端口 //将套接字文件描述符绑定到本地地址和端口
int err = bind(s, (struct sockaddr*)&local, sizeof(local)); //主处理过程
while()
{
memset(buff, , BUFFLEN);
n = recvfrom(s, buff, BUFFLEN,,(struct sockaddr*)&to, &len);//接收发送方数据
if(n > && !strncmp(buff, "TIME", ))//判断是否合法接收数据
{
printf("Get One Client Connect\n");
memset(buff, , BUFFLEN);
now = time(NULL);
sprintf(buff, "%24s\r\n",ctime(&now));
sendto(s, buff, strlen(buff),, (struct sockaddr*)&to, len);//发送数据
}
}
close(s); return ;
}
/** UDP循环服务器--client端程序**/
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <time.h>
#include <string.h>
#include <stdio.h>
#define BUFFLEN 1024
#define SERVER_PORT 12345
int main(int argc, char *argv[])
{
int s; //服务器套接字文件描述符
struct sockaddr_in server; //本地地址
time_t now;
char buff[BUFFLEN];
int n = ;
int len = ; //地址长度 //建立UDP套接字
s = socket(AF_INET, SOCK_DGRAM, ); //初始化地址接
memset(&server, , sizeof(server));
server.sin_family = AF_INET;//AF_INET协议族
server.sin_addr.s_addr = htonl(INADDR_ANY);//任意本地地址
server.sin_port = htons(SERVER_PORT);//服务器端口 memset(buff, , BUFFLEN);
strcpy(buff, "TIME");
//发送数据
sendto(s, buff, strlen(buff), , (struct sockaddr*)&server, sizeof(server));
memset(buff, , BUFFLEN);
//接收数据
len = sizeof(server);
n = recvfrom(s, buff, BUFFLEN, , (struct sockaddr*)&server, &len);
if(n >)
printf("TIME:%s",buff); close(s); return ;
}
TCP循环服务器
/** TCP循环服务器--server端程序**/
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <time.h>
#include <string.h>
#include <stdio.h>
#define BUFFLEN 1024
#define SERVER_PORT 12346
#define BACKLOG 5
int main(int argc, char *argv[])
{
int s_s, s_c; /*服务器套接字文件描述符*/
struct sockaddr_in local, from; /*本地地址*/
time_t now;
char buff[BUFFLEN];
int n = ;
int len = sizeof(from); /*建立TCP套接字*/
s_s = socket(AF_INET, SOCK_STREAM, ); /*初始化地址*/
memset(&local, , sizeof(local));
local.sin_family = AF_INET;/*AF_INET协议族*/
local.sin_addr.s_addr = htonl(INADDR_ANY);/*任意本地地址*/
local.sin_port = htons(SERVER_PORT);/*服务器端口*/ /*将套接字文件描述符绑定到本地地址和端口*/
int err = bind(s_s, (struct sockaddr*)&local, sizeof(local));
err = listen(s_s, BACKLOG);/*侦听*/ /*主处理过程*/
while()
{
/*接收客户端连接*/
s_c = accept(s_s, (struct sockaddr*)&from, &len);
memset(buff, , BUFFLEN);
n = recv(s_c, buff, BUFFLEN,);/*接收发送方数据*/
if(n > && !strncmp(buff, "TIME", ))/*判断是否合法接收数据*/
{
memset(buff, , BUFFLEN);
now = time(NULL);
sprintf(buff, "%24s\r\n",ctime(&now));
send(s_c, buff, strlen(buff),);/*发送数据*/
}
close(s_c);
}
close(s_s); return ;
}
/**TCP循环服务器--client端程序**/
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <string.h>
#include <stdio.h> #define BUFFLEN 1024
#define SERVER_PORT 12346
int main(int argc, char *argv[])
{
int s; /*服务器套接字文件描述符*/
struct sockaddr_in server; /*本地地址*/
char buff[BUFFLEN];
int n = ; /*建立TCP套接字*/
s = socket(AF_INET, SOCK_STREAM, ); /*初始化地址*/
memset(&server, , sizeof(server));
server.sin_family = AF_INET;/*AF_INET协议族*/
server.sin_addr.s_addr = htonl(INADDR_ANY);/*任意本地地址*/
server.sin_port = htons(SERVER_PORT);/*服务器端口*/ /*连接服务器*/
int err = connect(s, (struct sockaddr*)&server,sizeof(server));
memset(buff, , BUFFLEN);
strcpy(buff, "TIME");
/*发送数据*/
send(s, buff, strlen(buff), );
memset(buff, , BUFFLEN);
/*接收数据*/
n = recv(s, buff, BUFFLEN, );
if(n >){
printf("TIME:%s",buff);
}
close(s); return ;
}
两者返回给客户端的的输出都是一样的,比如:TIME:Sat Mar 22 15:26:25 2014
循环服务器的介绍就到这里。接下来介绍并发服务器模型。
Linux网络编程服务器模型选择之循环服务器的更多相关文章
- Linux网络编程服务器模型选择之并发服务器(上)
与循环服务器的串行处理不同,并发服务器对服务请求并发处理.循环服务器只能够一个一个的处理客户端的请求,显然效率很低.并发服务器通过建立多个子进程来实现对请求的并发处理.并发服务器的一个难点是如何确定子 ...
- Linux网络编程服务器模型选择之并发服务器(下)
前面两篇文章(参见)分别介绍了循环服务器和简单的并发服务器网络模型,我们已经知道循环服务器模型效率较低,同一时刻只能为一个客户端提供服务,而且对于TCP模型来说,还存在单客户端长久独占与服务器的连接, ...
- Linux网络编程服务器模型选择之IO复用循环并发服务器
在前面我们介绍了循环服务器,并发服务器模型.简单的循环服务器每次只能处理一个请求,即处理的请求是串行的,效率过低:并发服务器可以通过创建多个进程或者是线程来并发的处理多个请求.但是当客户端增加时,就需 ...
- Linux网络编程(简单的时间获取服务器)
1.一个简单的服务器时间获取程序 服务器和客户端采用UDP通信的方式,来编写一个简单的时间获取应用. 把过程大致理顺一下,首先是服务器端的编写,使用的是迭代的方式,没有并发 先创建一个socket而后 ...
- Linux网络编程(二)
Linux网络编程(二) 使用多进程实现服务器并发访问. 采用多进程的方式实现服务器的并发访问的经典范例. 程序实现功能: 1.客户端从标准输入读入一行文字,发送到服务器. 2.服务器接收到客户端发来 ...
- 服务器编程入门(4)Linux网络编程基础API
问题聚焦: 这节介绍的不仅是网络编程的几个API 更重要的是,探讨了Linux网络编程基础API与内核中TCP/IP协议族之间的关系. 这节主要介绍三个方面的内容:套接字( ...
- Linux 高性能服务器编程——Linux网络编程基础API
问题聚焦: 这节介绍的不仅是网络编程的几个API 更重要的是,探讨了Linux网络编程基础API与内核中TCP/IP协议族之间的关系. 这节主要介绍三个方面的内容:套接字(so ...
- Linux网络编程——tcp并发服务器(poll实现)
想详细彻底地了解poll或看懂下面的代码请参考<Linux网络编程——I/O复用之poll函数> 代码: #include <string.h> #include <st ...
- linux高性能服务器编程 (五) --Linux网络编程基础api
第五章 Linux网络编程基础api 1.主机字节序和网络字节序 字节序是指整数在内存中保存的顺序.字节序分为大端字节序.小端字节序. 大端字节序:一个整数的高位字节数据存放在内存的低地址处.低位字节 ...
随机推荐
- Redis数据结构(四)
存储list: list存储方式采用头和尾插入的方式,这样效率快,如果没有这个插入的数据,redis自己会创建这个数据,如果是中间插入的话,采用list方式效率就会很慢. ArrayList使用数组方 ...
- 前端福利之盘飞那个 "阿里矢量图" (转)
一.准备工作 1.首先,进入阿里的矢量图标库,在这个图标库里面可以找到很多图片资源,当然了需要登录才能下载或者使用,用GitHub账号或者新浪微博账号登录都可以 2.登录以后,可以搜索自己需要的资源, ...
- 安装完 swoole 后出现 PHP Warning: PHP Startup: Unable to load dynamic library 'swoole.so'的解决方法
安装完 swoole 后出现 PHP Warning: PHP Startup: Unable to load dynamic library 'swoole.so' (tried: /home/s ...
- hdu 4281 Judges' response(多旅行商&DP)
Judges' response Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
- CAS实战の自定义登录
由于每个版本的改动较大,所以先把版本号列出: 服务端版本:cas server 4.0.0 客户端版本:cas client 3.3.3 一.自定义登录页面 页面路径:/WebContent/WEB- ...
- mono+jexus 部署之CompilationException
使用 HelpPage 组件 CompilationException 好不容易在ubuntu上搭建了mono+jexus,欣喜若狂的部署上发布的网站,急忙打开,成功运行. 但是别高兴的太早,当我打开 ...
- Algebraic Foundations ( Arithmetic and Algebra) CGAL 4.13 -User Manual
理解: 本节主要介绍CGAL的代数结构和概念之间的互操作.与传统数论不同,CGAL的代数结构关注于实数轴的“可嵌入”特征.它没有将所有传统数的集合映射到自己的代数结构概念中,避免使用“数的类型”这一术 ...
- 知识记录——Session与Cookie
Session: Session是“会话”的意思,然而,因为http协议是无状态的,那么每次客户端请求服务器端,服务器端都会以“崭新”的页面展示给客户端,这在静态的html页面中是不会存在任何影响,但 ...
- pageadmin CMS网站建设教程:站点添加自定义字段
首先看看pagedmin默认的站点设置都有什么,如下图: 这里只有一些最基本的参数设置,用过3.0版本或用过其他公司开发的cms的用户应该有这种体验,在站点设置中可以设置logo图片,备案号,底部内容 ...
- docker 配置 overlay 存储
1.查看overlay模块是否安装 lsmod | grep over 2.将OverlayFS加到module目录下 echo "overlay" > /etc/modul ...