当前位置:   article > 正文

Libevent使用实例3 (高级) : evconnlistenner监听器_libevent evconnlistener 获得fd

libevent evconnlistener 获得fd

客户端:

  1. #include<sys/types.h>
  2. #include<sys/socket.h>
  3. #include<netinet/in.h>
  4. #include<arpa/inet.h>
  5. #include<errno.h>
  6. #include<unistd.h>
  7. #include<stdio.h>
  8. #include<string.h>
  9. #include<stdlib.h>
  10. #include<event.h>
  11. #include<event2/bufferevent.h>
  12. #include<event2/buffer.h>
  13. #include<event2/util.h>
  14. int tcp_connect_server(const char* server_ip, int port);
  15. void cmd_msg_cb(int fd, short events, void* arg);
  16. void server_msg_cb(struct bufferevent* bev, void*arg );
  17. void event_cb(struct bufferevent *bev, short event, void *arg);
  18. int main(int argc, char** argv) {
  19. if(argc < 3) {
  20. //两个参数依次是服务器端的IP地址,端口号
  21. printf("please input 2 parameter \n");
  22. return -1;
  23. }
  24. struct event_base *base = event_base_new();
  25. struct bufferevent* bev = bufferevent_socket_new(base, -1, BEV_OPT_CLOSE_ON_FREE);
  26. //监听终端输入事件
  27. struct event* ev_cmd = event_new(base, STDIN_FILENO, EV_READ | EV_PERSIST, cmd_msg_cb, (void*)bev);
  28. event_add(ev_cmd,NULL);
  29. struct sockaddr_in server_addr;
  30. memset(&server_addr,0,sizeof(server_addr) );
  31. server_addr.sin_family = AF_INET;
  32. server_addr.sin_port = htons(atoi(argv[2]));
  33. inet_aton(argv[1], &server_addr.sin_addr);
  34. bufferevent_socket_connect(bev, (struct sockaddr *)&server_addr,sizeof(server_addr));
  35. bufferevent_setcb(bev, server_msg_cb, NULL, event_cb, (void*)ev_cmd);
  36. bufferevent_enable(bev,EV_READ | EV_PERSIST);
  37. event_base_dispatch(base);
  38. printf("finished \n");
  39. return 0;
  40. }
  41. void cmd_msg_cb(int fd, short events, void* arg) { //fd:STDIN_FILENO, events:EV_READ | EV_PERSIST, arg:bev
  42. char msg[1024];
  43. int ret = read(fd, msg, sizeof(msg));
  44. if (ret < 0) {
  45. perror("read fail");
  46. exit(1);
  47. }
  48. struct bufferevent* bev = (struct bufferevent*)arg;
  49. //把终端的消息发送给服务器端
  50. bufferevent_write(bev,msg,ret);
  51. }
  52. void server_msg_cb(struct bufferevent* bev, void* arg) { //读取服务器发送的消息,打印到标准输出
  53. char msg[1024];
  54. size_t len = bufferevent_read(bev,msg,sizeof(msg));
  55. msg[len] = '\0';
  56. printf("recv %s from server \n",msg);
  57. }
  58. void event_cb(struct bufferevent* bev, short event, void *arg) {
  59. if (event & BEV_EVENT_EOF)
  60. printf("connection closed \n");
  61. else if(event & BEV_EVENT_ERROR)
  62. printf("some other error \n");
  63. else if(event & BEV_EVENT_CONNECTED)
  64. {
  65. printf("the client has connected to server \n");
  66. return ;
  67. }
  68. //这将自动close套接字和free读写缓冲区
  69. bufferevent_free(bev);
  70. struct event* ev = (struct event*) arg;
  71. event_free(ev);
  72. }

 

服务器端:

  1. #include<netinet/in.h>
  2. #include<sys/socket.h>
  3. #include<unistd.h>
  4. #include<stdio.h>
  5. #include<string.h>
  6. #include<event.h>
  7. #include<event2/listener.h>
  8. #include<event2/bufferevent.h>
  9. #include<event2/thread.h>
  10. void listener_cb(struct evconnlistener *listener, evutil_socket_t fd, struct sockaddr *sock, int socklen, void *arg);
  11. void socket_read_cb(struct bufferevent *bev, void *arg);
  12. void socket_event_cb(struct bufferevent *bev, short events, void *arg);
  13. int main()
  14. {
  15. //evthread_use_pthreads(); //enable threads
  16. struct sockaddr_in sin;
  17. memset(&sin, 0, sizeof(struct sockaddr_in));
  18. sin.sin_family = AF_INET;
  19. sin.sin_port = htons(9999);
  20. struct event_base *base = event_base_new();
  21. struct evconnlistener *listener = evconnlistener_new_bind(base, listener_cb, base,
  22. LEV_OPT_REUSEABLE | LEV_OPT_CLOSE_ON_FREE,
  23. 10, (struct sockaddr*)&sin, sizeof(struct sockaddr_in) );
  24. event_base_dispatch(base);
  25. evconnlistener_free(listener);
  26. event_base_free(base);
  27. return 0;
  28. }
  29. //一个新客户端连接上服务器了
  30. //当此函数被调用时,libevent已经帮我们accept了这个客户端,该客户端的文件描述符为fd
  31. void listener_cb(struct evconnlistener *listener, evutil_socket_t fd, struct sockaddr *sock, int socklen, void *arg) {
  32. printf("accept a client %d \n",fd);
  33. struct event_base *base = (struct event_base*)arg;
  34. //为这个客户端分配一个bufferevent
  35. struct bufferevent *bev = bufferevent_socket_new(base,fd,BEV_OPT_CLOSE_ON_FREE);
  36. bufferevent_setcb(bev,socket_read_cb,NULL,socket_event_cb,NULL);
  37. bufferevent_enable(bev,EV_READ | EV_PERSIST);
  38. }
  39. void socket_read_cb(struct bufferevent *bev, void *arg) { //bev:bev, arg:NULL
  40. char msg[4096];
  41. size_t len = bufferevent_read(bev,msg,sizeof(msg)-1);
  42. msg[len] = '\0';
  43. printf("server read the data %s \n",msg);
  44. char reply[] = "I has read your data";
  45. bufferevent_write(bev,reply,strlen(reply) ); //读取客户端发送的消息,并回复客户端
  46. }
  47. void socket_event_cb(struct bufferevent *bev, short events, void *arg) {
  48. if(events & BEV_EVENT_EOF)
  49. printf("connection closed \n");
  50. else if(events & BEV_EVENT_ERROR)
  51. printf("some other error \n");
  52. //这将自动close套接字和free读写缓冲区
  53. bufferevent_free(bev);
  54. }

服务器端使用evconnlistener监听器来接受客户端的连接,封装了连接过程。 用户只需初始化struct sockaddr_in结构体变量,然后把它作为参数传给函数evconnlistener_new_bind即可。

 

同样编译:

gcc -o xxx.c  xxx   -levent

 

运行结果:
客户端:

 

服务器端:

 

 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/IT小白/article/detail/923244?site
推荐阅读
相关标签
  

闽ICP备14008679号