用libev的c語言版本實現簡單的網路通訊伺服器

來源:互聯網
上載者:User

由於最近現網的epoll版本伺服器,出現了點詭異的問題,不得已改用libev快速上線,在這裡先記錄下簡單的使用執行個體。代碼中可能存在部分bug,此代碼並非線上跑的代碼,不過已經五髒俱全,如果有任何疑問,歡迎一起討論。

轉載註明出處:http://blog.csdn.net/lengzijian/article/details/8315133

#include <ev.h>#include <stdio.h>#include <netinet/in.h>#include<stdlib.h>#include <string.h>#define PORT 57789#define BUFFER_SIZE 2048#define MAX_ALLOWED_CLIENT 10240struct ev_io *libevlist[MAX_ALLOWED_CLIENT] = {NULL};void accept_cb(struct ev_loop *loop, struct ev_io *watcher, int revents);void read_cb(struct ev_loop *loop, struct ev_io *watcher, int revents);void timer_beat(struct ev_loop *loop, struct ev_timer *watcher, int revents);int freelibev(struct ev_loop *loop, int fd);int main(){    struct ev_loop *loop=ev_default_loop(0);    int sd;    struct sockaddr_in addr;    int addr_len = sizeof(addr);        //建立一個io watcher和一個timer watcher    struct ev_io socket_accept;    struct ev_timer timeout_w;    //建立socket串連    sd = socket(PF_INET, SOCK_STREAM, 0);    if(sd < 0)    {        printf("socket error\n");        return -1;    }    bzero(&addr, sizeof(addr));    addr.sin_family = AF_INET;    addr.sin_port = htons(PORT);    addr.sin_addr.s_addr = INADDR_ANY;    //正常bind    if(bind(sd, (struct sockaddr*)&addr, sizeof(addr)) != 0)    {        printf("bind error\n");        return -1;    }    //正常listen    if(listen(sd, SOMAXCONN) < 0)    {        printf("listen error\n");        return -1;    }    //設定fd可重用    int bReuseaddr=1;    if(setsockopt(sd,SOL_SOCKET ,SO_REUSEADDR,(const char*)&bReuseaddr,sizeof(bReuseaddr)) != 0)    {        printf("setsockopt error in reuseaddr[%d]\n", sd);        return ;    }        //初始化io watcher,用於監聽fd    ev_io_init(&socket_accept, accept_cb, sd, EV_READ);    ev_io_start(loop, &socket_accept);        //可以向遠端發送心跳包    //ev_timer_init(&timeout_w, timer_beat, 1., 0.);    //ev_timer_start(loop, &timeout_w);        ev_run(loop, 0);        return 1;}void accept_cb(struct ev_loop *loop, struct ev_io *watcher, int revents){    /*        如果有連結,則繼續監聽fd;    */    struct sockaddr_in client_addr;    socklen_t client_len = sizeof(client_addr);    int client_sd;    //建立用戶端的io watcher    struct ev_io *w_client = (struct ev_io*) malloc(sizeof(struct ev_io));        if(w_client == NULL)    {        printf("malloc error in accept_cb\n");        return ;    }        if(EV_ERROR & revents)    {        printf("error event in accept\n");        return ;    }        //擷取與用戶端相連的fd    client_sd = accept(watcher->fd, (struct sockaddr*)&client_addr, &client_len);    if(client_sd < 0)    {        printf("accept error\n");        return;    }    //如果串連數超出指定範圍,則關閉串連    if( client_sd > MAX_ALLOWED_CLIENT)    {        printf("fd too large[%d]\n", client_sd);        close(client_sd);        return ;    }        if(libevlist[client_sd] != NULL)    {        printf("client_sd not NULL fd is [%d]\n", client_sd);        return ;    }        printf("client connected\n");    //監聽新的fd    ev_io_init(w_client, read_cb, client_sd, EV_READ);    ev_io_start(loop, w_client);        libevlist[client_sd] = w_client;}void read_cb(struct ev_loop *loop, struct ev_io *watcher, int revents){    char buffer[BUFFER_SIZE];    ssize_t read;        if(EV_ERROR & revents)    {        printf("error event in read\n");        return ;    }    //正常的recv    read = recv(watcher->fd, buffer, BUFFER_SIZE, 0);    if(read < 0)    {        printf("read error\n");        return;    }        if(read == 0)    {        printf("client disconnected.\n");        //ev_io_stop(loop, watcher);        //free(watcher);        //如果用戶端中斷連線,釋放響應的資源,並且關閉監聽        freelibev(loop, watcher->fd);        return;    }    else    {        //buffer[read] = '\0';        printf("receive message:%s\n", buffer);    }    //返回給用戶端    send(watcher->fd, buffer, read, 0);    bzero(buffer, read);}void timer_beat(struct ev_loop *loop, struct ev_timer *watcher, int revents){    float timeout = 2.0;    //這裡可以發送心跳包,也可以什麼都不做    printf("send beat per[%f]\n",timeout);        fflush(stdout);        if(EV_ERROR & revents)    {        printf("error event in timer_beat\n");        return ;    }        ev_timer_set(watcher, timeout,0.);    ev_timer_start(loop, watcher);    return;}int freelibev(struct ev_loop *loop, int fd){    /*if(fd > MAX_ALLOWED_CLIENT)    {        printf("more than MAX_ALLOWED_CLIENT[%d]", fd);        return -1;    }*/    //清理相關資源        if(libevlist[fd] == NULL)    {        printf("the fd already freed[%d]\n", fd);        return -1;    }        close(fd);    ev_io_stop(loop, libevlist[fd]);    free(libevlist[fd]);    libevlist[fd] = NULL;    return 1;}


 

makefile:

CC=gccFLAGS=-I. -I/home/lengzijian/c/libev/libev-4.11 LDFLAGS=-L/usr/lib -L/home/lengzijian/c/libev/libev-4.11/.libs -lev -WallOBJECTS=server.oALL_BIN=serverall:$(ALL_BIN) $(ALL_BIN):$(OBJECTS) $(CC) $(FLAGS) $(LDFLAGS) -o $@ $^%.o:%.c$(CC) -c $< $(FLAGS) $(FLAGS)clean:rm -fr $(OBJECTS)rm -fr $(ALL_BIN)


 

注意makefile中的標頭檔路徑,和動態連結程式庫的路徑

測試時,只需要telnet ip 57789 即可

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.