基于select的服务端

转载 2016年05月31日 10:58:54

 * 1. 程序使用了一个数组fd,通信开始后把需要通信的多个socket描述符都放入此数组
 * 2. 首先生成一个叫sock_fd的socket描述符,用于监听端口。
 * 3. 将sock_fd和数组fd中不为0的描述符放入select将检查的集合fdsr。
// select_server.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/time.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#define MYPORT 9999 //连接时使用的端口
#define MAXCLINE 5 //连接队列中的个数
#define BUF_SIZE 200
int fd[MAXCLINE]; //连接的fd
int conn_amount; //当前的连接数
void showclient()
{
    int i;
    printf("client amount:%d\n",conn_amount);
    for(i=0;i<MAXCLINE;i++)
    {
        printf("[%d]:%d ",i,fd[i]);
    }
    printf("\n\n");
}
int main(void)
{
    int sock_fd,new_fd; //监听套接字 连接套接字
    struct sockaddr_in server_addr; // 服务器的地址信息
    struct sockaddr_in client_addr; //客户端的地址信息
    socklen_t sin_size;
    int yes = 1;
    char buf[BUF_SIZE];
    int ret;
    int i;
    //建立sock_fd套接字
    if((sock_fd = socket(AF_INET,SOCK_STREAM,0))==-1)
    {
        perror("setsockopt");
        exit(1);
    }
    //设置套接口的选项 SO_REUSEADDR 允许在同一个端口启动服务器的多个实例
    // setsockopt的第二个参数SOL SOCKET 指定系统中,解释选项的级别 普通套接字
    if(setsockopt(sock_fd,SOL_SOCKET,SO_REUSEADDR,&yes,sizeof(int))==-1)
    {
        perror("setsockopt error \n");
        exit(1);
    }
    server_addr.sin_family = AF_INET; //主机字节序
    server_addr.sin_port = htons(MYPORT);
    server_addr.sin_addr.s_addr = INADDR_ANY;//通配IP
    memset(server_addr.sin_zero,'\0',sizeof(server_addr.sin_zero));
    if(bind(sock_fd,(struct sockaddr *)&server_addr,sizeof(server_addr)) == -1)
    {
        perror("bind error!\n");
        exit(1);
    }
    if(listen(sock_fd,MAXCLINE)==-1)
    {
        perror("listen error!\n");
        exit(1);
    }
    printf("listen port %d\n",MYPORT);
    fd_set fdsr; //文件描述符集的定义
    int maxsock;
    struct timeval tv;
    conn_amount =0;
    sin_size = sizeof(client_addr);
    maxsock = sock_fd;
    while(1)
    {
    //初始化文件描述符集合
    FD_ZERO(&fdsr); //清除描述符集
    FD_SET(sock_fd,&fdsr); //把sock_fd加入描述符集
    //超时的设定
    tv.tv_sec = 30;
    tv.tv_usec =0;
    //添加活动的连接
    for(i=0;i<MAXCLINE;i++)
    {
        if(fd[i]!=0)
        {
            FD_SET(fd[i],&fdsr);
        }
    }
    //如果文件描述符中有连接请求 会做相应的处理,实现I/O的复用 多用户的连接通讯
    ret = select(maxsock +1,&fdsr,NULL,NULL,&tv);
    if(ret <0) //没有找到有效的连接 失败
    {
        perror("select error!\n");
        break;
    {
        printf("timeout \n");
        continue;
    }
    //循环判断有效的连接是否有数据到达
    for(i=0;i<conn_amount;i++)
    {
        if(FD_ISSET(fd[i],&fdsr))
        {
            ret = recv(fd[i],buf,sizeof(buf),0);
            if(ret <=0) //客户端连接关闭,清除文件描述符集中的相应的位
                close(fd[i]);
                FD_CLR(fd[i],&fdsr);
                fd[i]=0;
                conn_amount--;
            }
                //否则有相应的数据发送过来 ,进行相应的处理
            else
            {
                if(ret <BUF_SIZE)
                memset(&buf[ret],'\0',1);
                printf("client[%d] send:%s\n",i,buf);
            }
        }       
    }
    if(FD_ISSET(sock_fd,&fdsr))
    {
        new_fd = accept(sock_fd,(struct sockaddr *)&client_addr,&sin_size);
        if(new_fd <=0)
        {
            perror("accept error\n");
            continue;
        }
        if(conn_amount <MAXCLINE)
        {   
            for(i=0;i< MAXCLINE;i++)
            {
                if(fd[i]==0)
                {
                    fd[i] = new_fd;
                    break;
                }               
            }
            conn_amount++;
            printf("new connection client[%d]%s:%d\n",conn_amount,inet_ntoa(client_addr.sin_addr),n
tohs(client_addr.sin_port));
            if(new_fd > maxsock)
            {
                maxsock = new_fd;
            }               
        }
        else
        {
            printf("max connections arrive ,exit\n");
            send(new_fd,"bye",4,0);
            close(new_fd);
            continue;
        }
    }
    showclient();
}
    for(i=0;i<MAXCLINE;i++)
    {
        if(fd[i]!=0)
        {
            close(fd[i]);
        }
    }
    exit(0);





//使用select注意事项:

1.要将sock_fd加入到maxfd+1中,要不就无法检测到网络连接,会一直阻塞在select语句

2.通过存储每次连接的描述符,设置FD_SET函数,在遍历的去判断FD_ISSET处理。

3.我们可以看到select每次有数据到来时,需要遍历的去寻找所有可用的描述符,来判断其是否满足处理的条件。

4.select的通知机制,轮询的去查看是否在maxfd+1内有满足条件的描述符。




I/O复用Select函数的UDP和TCP客户端和服务端

I/O复用Select函数的UDP和TCP客户端和服务端 IO复用Select函数的UDP和TCP客户端和服务端  要求  设计思路  实验结果  基本的时间传输 TCP  UDP  体会  代码  ...
  • Volcano3511
  • Volcano3511
  • 2016年12月20日 20:01
  • 658

基于select模型的tcp服务器------一个服务器如何与多个客户端进行通信?

很多时候, 服务器都需要同时与多个客户端进行通信, 服务嘛, 就是这样。 下面, 我们用select模型来简要模拟一下这种情形。代码是最好的解释, 所以, 还是上代码吧:       服务端程序: #...
  • stpeace
  • stpeace
  • 2015年03月17日 22:00
  • 7595

深入研究socket编程(3)——使用select函数编写客户端和服务器

首先看原先《UNIX网络编程——并发服务器(TCP)》的代码,服务器代码serv.c: [cpp] view plaincopyprint? #include   #inclu...
  • chenxun2009
  • chenxun2009
  • 2016年01月09日 16:26
  • 3054

linux网络编程之socket(九):使用select函数改进客户端/服务器端程序

一、当我们使用单进程单连接且使用readline修改后的客户端程序,去连接使用readline修改后的服务器端程序,会出现一个有趣的现象,先来看输出: 先运行服务器端,再运行客户端, simba@ub...
  • Simba888888
  • Simba888888
  • 2013年06月10日 19:45
  • 15231

unix的select实现服务端与多客户端通讯

  • 2015年06月02日 23:00
  • 89KB
  • 下载

Linux下网络socket编程——实现服务器(select)与多个客户端通信

一、关于socket通信服务器端工作流程: 调用 socket() 函数创建套接字 用 bind() 函数将创建的套接字与服务端IP地址绑定 调用listen()函数监听socket() 函数创建的套...
  • qicheng777
  • qicheng777
  • 2017年06月23日 14:44
  • 941

基于select模型的TCP服务器

之前的一篇博文是基于TCP的服务器和客户机程序,今天在这我要实现一个基于select模型的TCP服务器(仅实现了服务器)。 socket套接字编程提供了很多模型来使服务器高效的接受客户端的请求,se...
  • leex_brave
  • leex_brave
  • 2016年07月29日 20:36
  • 2685

Linux select服务器

select原理系统提供select函数来实现多路复用输入/输出模型。select系统调用是用来让我们的程序监视多个文件句柄的状态变化的。程序会停在select这里等待,直到被监视的文件句柄有一个或多...
  • Dakuan_chen
  • Dakuan_chen
  • 2017年06月28日 22:28
  • 188

采用select实现多客户服务端

服务端用fork来处理多个客户的思路是为每个客户创建一个新建的进程进行单独处理,也就是采取了多个服务器进程的方式,这在涉及到数据库的应用中不是最佳的解决方案。一般的解决方案是让单个服务器进程在不阻塞、...
  • gigglesun
  • gigglesun
  • 2013年10月20日 11:06
  • 1201

理解 select 函数并实现服务器端

运用select 函数是最具有代表性的实现复用服务器端方法。Windows平台下也有同名函数提供相同功能,因此具有良好的移植性。     select函数的功能和调用顺序     使用select...
  • u012469252
  • u012469252
  • 2015年01月24日 16:40
  • 853
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:基于select的服务端
举报原因:
原因补充:

(最多只允许输入30个字)