linux之I/O复用

//utili.h
#pragma once

#include<stdio.h>
#include<unistd.h>
#include<string.h>
#include<stdlib.h>
#include<sys/socket.h>
#include<netinet/in.h>
#include<arpa/inet.h>

#include<pthread.h>

#define BUFFER_SIZE 256
#define LISTEN_QUEUE_SIZE 5


int startup(char *ip, short port)
{
    int sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if(sockfd == -1)
    {
        perror("socket");
        return -1;
    }
    struct sockaddr_in address;
    address.sin_family = AF_INET;
    address.sin_port = htons(port);
    address.sin_addr.s_addr = inet_addr(ip);

    int yes = 1;
    setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(int));

    socklen_t addrlen = sizeof(struct sockaddr);
    bind(sockfd, (struct sockaddr*)&address, addrlen);
    listen(sockfd, LISTEN_QUEUE_SIZE);
    return sockfd;
}





//cli.c
#include"utili.h"

int main(int argc, char *argv[])
{
    int sockCli = socket(AF_INET, SOCK_STREAM, 0);
    if(sockCli == -1)
    {
        perror("socket");
        return -1;
    }

    struct sockaddr_in addrSer;
    addrSer.sin_family = AF_INET;
    addrSer.sin_port = htons(atoi(argv[2]));
    addrSer.sin_addr.s_addr = inet_addr(argv[1]);

    socklen_t addrlen = sizeof(struct sockaddr);
    int ret = connect(sockCli, (struct sockaddr*)&addrSer, addrlen);
    if(ret == -1)
    {
        perror("connect");
        return -1;
    }

    char buffer[256];
    char recvbuf[256];
    while(1)
    {
        printf("Cli:>");
        scanf("%s",buffer);
        send(sockCli, buffer, strlen(buffer)+1, 0);
        recv(sockCli, recvbuf, 256, 0);
        printf("Form Server self msg:> %s\n",recvbuf);
    }

    close(sockCli);
    return 0;
}

//select.c
#include"../utili.h"
#include<sys/select.h>

#define MAX_CLIENT_SIZE 5

void data_handler(int sockConn)
{
    char buffer[256];
    recv(sockConn, buffer, 256, 0);
    printf("cli msg:> %s\n",buffer);
    send(sockConn, buffer, strlen(buffer)+1, 0);
}

//   ./ser 127.0.0.1 8080

int main(int argc, char *argv[])
{
    int sockSer = startup(argv[1], atoi(argv[2]));

    //1 accept
    //2 data_handler

    fd_set set;
    int max_sock = sockSer;
    int client_fd[MAX_CLIENT_SIZE] = {0};
    int client_conn_num = 0;
    int i;
    while(1)
    {
        FD_ZERO(&set);
        FD_SET(sockSer, &set);
        for(i=0; i<MAX_CLIENT_SIZE; ++i)
        {
            if(client_fd[i] != 0)
            {
                FD_SET(client_fd[i], &set);
            }
        }

        struct timeval tv;
        tv.tv_sec = 5;
        tv.tv_usec = 0;
        int ret = select(max_sock+1, &set, NULL, NULL, &tv);
        if(ret == -1)
        {
            perror("select");
            continue;
        }
        else if(ret == 0)
        {
            printf("Server Time Out.\n");
            continue;
        }
        else
        {
            if(FD_ISSET(sockSer, &set))
            {
                struct sockaddr_in addrCli;
                socklen_t addrlen = sizeof(struct sockaddr);
                int sockConn = accept(sockSer, (struct sockaddr*)&addrCli, &addrlen);
                if(sockConn == -1)
                {
                    perror("accept");
                    continue;
                }
                if(client_conn_num >= MAX_CLIENT_SIZE)
                {
                    printf("Server Over Load.\n");
                    continue;
                }
                
                client_fd[client_conn_num++] =  sockConn;
                if(sockConn > max_sock)
                    max_sock = sockConn;
            }
            else
            {
                for(i=0; i<MAX_CLIENT_SIZE; ++i)
                {
                    if(FD_ISSET(client_fd[i], &set))
                    {
                        data_handler(client_fd[i]);
                    }
                }
            }
        }
    }

    close(sockSer);
    return 0;
}







//poll.c
#include"../utili.h"
#include<poll.h>

#define MAX_CLIENT_SIZE 5

void data_handler(int sockConn)
{
    char buffer[256];
    recv(sockConn, buffer, 256, 0);
    printf("from client msg:> %s\n",buffer);
    send(sockConn, buffer, strlen( buffer)+1, 0);
}

int main(int argc, char *argv[])
{
    int sockSer = startup(argv[1], atoi(argv[2]));

    struct pollfd client_fd[MAX_CLIENT_SIZE+1];
    client_fd[0].fd = sockSer;
    client_fd[0].events = POLLIN;

    int client_conn_num = 0;
    int i;
    while(1)
    {
        //1 accept
        int ret = poll(client_fd, client_conn_num+1, -1);
        if(ret == -1)
        {
            perror("poll");
            continue;
        }
        else if(ret == 0)
        {
            printf("Server Time Out.\n");
            continue;
        }
        else
        {
            if(client_fd[0].fd==sockSer && client_fd[0].revents==POLLIN)
            {
                struct sockaddr_in addrCli;
                socklen_t addrlen = sizeof(struct sockaddr);
                int sockConn = accept(sockSer, (struct sockaddr*)&addrCli, &addrlen);
                if(sockConn == -1)
                {
                    perror("accpet");
                    continue;
                }

                if(client_conn_num >= MAX_CLIENT_SIZE)
                {
                    char *msg = "Server Over Load.";
                    send(sockConn, msg, strlen(msg)+1, 0);
                    continue;
                }
                client_conn_num++;
                client_fd[client_conn_num].fd = sockConn;
                client_fd[client_conn_num].events = POLLIN;
            }
            else
            {
                for(i=1; i<MAX_CLIENT_SIZE; ++i)
                {
                    if(client_fd[i].revents & POLLIN)
                    {
                        data_handler(client_fd[i].fd);
                    }
                }
            }
        }
        //2 data_handler
    }

    close(sockSer);
    return 0;
}









//epoll.c
#include"../utili.h"
#include<sys/epoll.h>

#define MAX_CLIENT_SIZE 5
#define EPOLL_SIZE 1024

void data_handler(int sockConn)
{
    char buffer[256];
    recv(sockConn, buffer, 256, 0);
    printf("from client msg:> %s\n",buffer);
    send(sockConn, buffer, strlen( buffer)+1, 0);
}

void add_event(int epfd, int fd, int event)
{
    struct epoll_event ev;
    ev.events = event;
    ev.data.fd = fd;
    epoll_ctl(epfd, EPOLL_CTL_ADD, fd, &ev);
}
void mod_event(int epfd, int fd, int event)
{
     struct epoll_event ev;
    ev.events = event;
    ev.data.fd = fd;
    epoll_ctl(epfd, EPOLL_CTL_MOD, fd, &ev);
}
void del_event(int epfd, int fd, int event)
{
    struct epoll_event ev;
    ev.events = event;
    ev.data.fd = fd;
    epoll_ctl(epfd, EPOLL_CTL_DEL, fd, &ev);
}

int main(int argc, char *argv[])
{
    int sockSer = startup(argv[1], atoi(argv[2]));

    int epfd = epoll_create(EPOLL_SIZE);

    //add server event
    //struct epoll_event ev;
    //ev.events = EPOLLIN;
    //ev.data.fd = sockSer;
    //epoll_ctl(epfd, EPOLL_CTL_ADD, sockSer, &ev);
    add_event(epfd, sockSer, EPOLLIN);
    
    int i;
    struct epoll_event revent[MAX_CLIENT_SIZE];
    int client_conn_num = 0;
    while(1)
    {
        int ret = epoll_wait(epfd, revent, MAX_CLIENT_SIZE, -1);
        if(ret == -1)
        {
            perror("poll");
            continue;
        }
        else if(ret == 0)
        {
            printf("Server Time Out.\n");
            continue;
        }
        else
        {
            for(i=0; i<ret; ++i)
            {
                if(revent[i].data.fd==sockSer && (revent[i].events&EPOLLIN))
                {
                    struct sockaddr_in addrCli;
                    socklen_t addrlen = sizeof(struct sockaddr);
                    int sockConn = accept(sockSer, (struct sockaddr*)&addrCli, &addrlen);
                    if(sockConn == -1)
                    {
                        perror("accept");
                        continue;
                    }
                    if(client_conn_num >= MAX_CLIENT_SIZE)
                    {
                        char *msg = "Server Over Load.";
                        send(sockConn, msg, strlen(msg)+1, 0);
                        continue;
                    }
                    client_conn_num++;

                    //ev.events = EPOLLIN;
                    //ev.data.fd = sockConn;
                    //epoll_ctl(epfd, EPOLL_CTL_ADD, sockConn, &ev);
                    add_event(epfd, sockConn, EPOLLIN);
                }
                else if(revent[i].events & EPOLLIN)
                {
                    data_handler(revent[i].data.fd);
                }
            }
        }
    }

    close(sockSer);
    return 0;
}







参考资料:《linux高性能服务编程》第九章



  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值