libev实现的tiny socket server

转自http://blog.csdn.net/wusuopubupt/article/details/39055397

主要参考了3处:

1. libev官方手册: http://pod.tst.eu/http://cvs.schmorp.de/libev/ev.pod

2.lengzijian的博客: http://blog.csdn.net/lengzijian/article/details/8315133

3.公司自己的项目

----

[cpp]  view plain  copy
  1. /* 
  2.  * @author  : wusuopubupt 
  3.  * @date    : 2014-09-04 
  4.  * @desc    : tiny socket server implemented by libev 
  5.  *            to use this, you should install libev at first. 
  6.  * 
  7.  *            server: just run the program 
  8.  *            client: telnet localhost 8080 
  9.  * 
  10.  * @refer   : 1). http://pod.tst.eu/http://cvs.schmorp.de/libev/ev.pod 
  11.  *        2). http://blog.csdn.net/lengzijian/article/details/8315133 
  12.  * 
  13.  */  
  14. #include <ev.h>  
  15. #include <stdio.h>  
  16. #include <stdlib.h>  
  17. #include <string.h>  
  18. #include <netinet/in.h>  
  19. #include <unistd.h>  
  20.   
  21. #define PORT 8080  
  22. #define BUFFER_SIZE 1024  
  23. #define MAX_CONNECTIONS 10  
  24.   
  25. struct ev_io *libevlist[MAX_CONNECTIONS] = {NULL};  
  26.   
  27. void socket_accept_callback(struct ev_loop *loop, struct ev_io *watcher, int revents);  
  28. void socket_read_callback(struct ev_loop *loop, struct ev_io *watcher, int revents);  
  29. /* 
  30.     Server                  Client 
  31.  
  32.  
  33.     socket                  socket 
  34.       |                       | 
  35.       v                       v 
  36.     bind                    connect 
  37.       |                       | 
  38.       v                       v 
  39.     listen                  write 
  40.       |                       | 
  41.       v                       v 
  42.     accept                  read 
  43.       |                       | 
  44.       v                       v 
  45.     read                    close 
  46.       | 
  47.       v 
  48.     write 
  49.       | 
  50.       v 
  51.     close 
  52. */  
  53.   
  54. int main() {  
  55.     struct ev_loop *loop = ev_default_loop(0);  
  56.   
  57.     /* socket start */  
  58.     int sd;  
  59.     struct sockaddr_in addr;  
  60.     int addr_len = sizeof(addr);  
  61.   
  62.     struct ev_io *socket_watcher = (struct ev_io*)malloc(sizeof(struct ev_io));  
  63.     struct ev_timer *timeout_watcher = (struct ev_timer*)malloc(sizeof(struct ev_timer));  
  64.   
  65.     // socket  
  66.     sd = socket(PF_INET, SOCK_STREAM, 0);  
  67.     if (sd < 0) {  
  68.         printf("socket error\n");  
  69.         return -1;  
  70.     }  
  71.     bzero(&addr, sizeof(addr));  
  72.     addr.sin_family = AF_INET;  
  73.     addr.sin_port = htons(PORT);  
  74.     addr.sin_addr.s_addr = INADDR_ANY;  
  75.   
  76.     // bind  
  77.     if (bind(sd, (struct sockaddr*) &addr, sizeof(addr)) != 0) {  
  78.         printf("bind error\n");  
  79.         return -1;  
  80.     }  
  81.     // listen  
  82.     if (listen(sd, SOMAXCONN) < 0) {  
  83.         printf("listen error\n");  
  84.         return -1;  
  85.     }  
  86.     // set sd reuseful  
  87.     int bReuseaddr = 1;  
  88.     if (setsockopt(sd, SOL_SOCKET, SO_REUSEADDR, (const char*) &bReuseaddr, sizeof(bReuseaddr)) != 0) {  
  89.         printf("setsockopt error in reuseaddr[%d]\n", sd);  
  90.         return -1;  
  91.     }  
  92.     /* socket end */  
  93.   
  94.     ev_io_init(socket_watcher, socket_accept_callback, sd, EV_READ);  
  95.     ev_io_start(loop, socket_watcher);  
  96.   
  97.     while(1) {  
  98.         ev_run(loop, 0);  
  99.     }  
  100.   
  101.     return 1;  
  102. }  
  103.   
  104. void socket_accept_callback(struct ev_loop *loop, struct ev_io *watcher, int revents) {  
  105.     printf("I am: %d\n", getpid());  
  106.   
  107.     struct sockaddr_in client_addr;  
  108.     socklen_t client_len = sizeof(client_addr);  
  109.     int client_sd;  
  110.   
  111.     // ev_io watcher for client  
  112.     struct ev_io *client_watcher = (struct ev_io*) malloc(sizeof(struct ev_io));  
  113.   
  114.     if (client_watcher == NULL) {  
  115.         printf("malloc error in accept_cb\n");  
  116.         return;  
  117.     }  
  118.   
  119.     if (EV_ERROR & revents) {  
  120.         printf("error event in accept\n");  
  121.         return;  
  122.     }  
  123.   
  124.     // socket accept: get file description  
  125.     client_sd = accept(watcher->fd, (struct sockaddr*) &client_addr, &client_len);  
  126.     if (client_sd < 0) {  
  127.         printf("accept error\n");  
  128.         return;  
  129.     }  
  130.     // too much connections  
  131.     if (client_sd > MAX_CONNECTIONS) {  
  132.         printf("fd too large[%d]\n", client_sd);  
  133.         close(client_sd);  
  134.         return;  
  135.     }  
  136.   
  137.     if (libevlist[client_sd] != NULL) {  
  138.         printf("client_sd not NULL fd is [%d]\n", client_sd);  
  139.         return;  
  140.     }  
  141.   
  142.     printf("client connected\n");  
  143.   
  144.     // listen new client  
  145.     ev_io_init(client_watcher, socket_read_callback, client_sd, EV_READ);  
  146.     ev_io_start(loop, client_watcher);  
  147.   
  148.     libevlist[client_sd] = client_watcher;  
  149. }  
  150.   
  151. void socket_read_callback(struct ev_loop *loop, struct ev_io *watcher, int revents) {  
  152.     char buffer[BUFFER_SIZE];  
  153.     ssize_t read;  
  154.   
  155.     if (EV_ERROR & revents) {  
  156.         printf("error event in read\n");  
  157.         return;  
  158.     }  
  159.     // socket recv  
  160.     read = recv(watcher->fd, buffer, BUFFER_SIZE, 0); // read stream to buffer  
  161.     if (read < 0) {  
  162.         printf("read error\n");  
  163.         return;  
  164.     }  
  165.   
  166.     if (read == 0) {  
  167.         printf("client disconnected.\n");  
  168.   
  169.         if (libevlist[watcher->fd] == NULL) {  
  170.             printf("the fd already freed[%d]\n", watcher->fd);  
  171.         }  
  172.         else {  
  173.             close(watcher->fd);  
  174.             ev_io_stop(loop, libevlist[watcher->fd]);  
  175.             free(libevlist[watcher->fd]);  
  176.             libevlist[watcher->fd] = NULL;  
  177.         }  
  178.         return;  
  179.     }  
  180.     else {  
  181.         printf("receive message:%s\n", buffer);  
  182.     }  
  183.   
  184.     // socket send to client  
  185.     send(watcher->fd, buffer, read, 0);  
  186.     bzero(buffer, read);  
  187. }  

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值