Windows下实现简单的libevent服务器_服务器其它

最近再学习Libevent由于自己使用的是windows系统,遗憾的是有关在vs下可以参考的程序少之又少。在参考了许多的博客文章后。自己摸索写了一个简单的Libevent Server程序。并且在网上找了一个简单的客户端程序,测试该代码成功。今天在此做一个记录。

Libevent的确是一个非常好用的东西,还在继续学习中,后续还要在windows下实现Libevent的多线程使用。今天先把自己搞出来的东西贴上来,仅供学习参考。在vs2015上编译通过。

默认情况下是单线程的(可以配置成多线程,如果有需要的话),每个线程有且只有一event base,对应一个struct event_base结构体(以及附于其上的事件管理器),用来schedule托管给它的一系列event,可以和操作系统的进程管理类比,当然,要更简单一点。当一个事件发生后,event_base会在合适的时间(不一定是立即)去调用绑定在这个事件上的函数(传入一些预定义的参数,以及在绑定时指定的一个参数),直到这个函数执行完,再返回schedule其他事件。

 //创建一个event_base
struct event_base *base = event_base_new();
assert(base != NULL); 

event_base内部有一个循环,循环阻塞在epoll / kqueue等系统调用上,直到有一个 / 一些事件发生,然后去处理这些事件。当然,这些事件要被绑定在这个event_base上。每个事件对应一个struct event,可以是监听一个fd或者POSIX信号量之类(这里只讲fd了,其他的看manual吧)。struct event使用event_new来创建和绑定,使用event_add来启用:

 //创建并绑定一个event
struct event *listen_event;
//参数:event_base, 监听的fd,事件类型及属性,绑定的回调函数,给回调函数的参数

listen_event = event_new(base, listener, EV_READ | EV_PERSIST, callback_func, (void*)base);
//参数:event,超时时间(struct timeval *类型的,NULL表示无超时设置)
event_add(listen_event, NULL);

注:libevent支持的事件及属性包括(使用bitfield实现,所以要用 | 来让它们合体)

(a) EV_TIMEOUT: 超时
(b) EV_READ : 只要网络缓冲中还有数据,回调函数就会被触发
(c) EV_WRITE : 只要塞给网络缓冲的数据被写完,回调函数就会被触发
(d) EV_SIGNAL : POSIX信号量,参考manual吧
(e) EV_PERSIST : 不指定这个属性的话,回调函数被触发后事件会被删除
(f) EV_ET : Edge - Trigger边缘触发,参考EPOLL_ET
然后需要启动event_base的循环,这样才能开始处理发生的事件。循环的启动event base dispatch,循环将一直持续,直到不再有需要关注的事件,或者是遇到event_loopbreak() / event_loopexit()函数。
//启动事件循环
event_base_dispatch(base);

接下来关注下绑定到event的回调函数callback_func:传递给它的是一个socket fd、一个event类型及属性bit_field、以及传递给event_new的最后一个参数(去上面几行回顾一下,把event_base给传进来了,实际上更多地是分配一个结构体,把相关的数据都撂进去,然后丢给event_new,在这里就能取得到了)。其原型是:
typedef void(*event_callback_fn)(evutil_socket_t sockfd, short event_type, void *arg) 

对于一个服务器而言,上面的流程大概是这样组合的:
1. listener = socket(),bind(),listen(),设置nonblocking(POSIX系统中可使用fcntl设置,windows不需要设置,
    实际上libevent提供了统一的包装evutil_make_socket_nonblocking)
2. 创建一个event_base
3. 创建一个event,将该socket托管给event_base,指定要监听的事件类型,并绑定上相应的回调函数(及需要给它的参数)
    。对于listener socket来说,只需要监听EV_READ | EV_PERSIST
4. 启用该事件
5. 进入事件循环
-------------- -
6. (异步)当有client发起请求的时候,调用该回调函数,进行处理。
/*接下来关注下绑定到event的回调函数callback_func:传递给它的是一个socket fd、一个event类型及属性bit_field、以及传递给event_new的最后一个参数(去上面几行回顾一下,把event_base给传进来了,实际上更多地是分配一个结构体,把相关的数据都撂进去,然后丢给event_new,在这里就能取得到了)。*/ 

服务器端代码:Server.cpp

  #include <WinSock2.h>
  #include <stdlib.h>
  #include <stdio.h>
  #include <string.h>
  #include <errno.h>
  #include <event2/event.h>
  #include <event2/bufferevent.h>
  #include<iostream>
  #include<cassert>
  #pragma comment (lib,"ws2_32.lib")
  #include<ws2tcpip.h>
  #define LISTEN_PORT 9999
  #define LIATEN_BACKLOG 32
 using namespace std;
 /*********************************************************************************
 *                   函数声明
 **********************************************************************************/
 //accept回掉函数
 void do_accept_cb(evutil_socket_t listener, short event, void *arg);
 //read 回调函数
 void read_cb(struct bufferevent *bev, void *arg);
 //error回调函数
 void error_cb(struct bufferevent *bev, short event, void *arg);
 //write 回调函数
 void write_cb(struct bufferevent *bev, void *arg);
 /*********************************************************************************
 *                   函数体
 **********************************************************************************/
 //accept回掉函数
 void do_accept_cb(evutil_socket_t listener, short event, void *arg)
 {
   //传入的event_base指针
   struct event_base *base = (struct event_base*)arg;
   //socket描述符
   evutil_socket_t fd;
   //声明地址
   struct sockaddr_in sin;
   //地址长度声明
   socklen_t slen = sizeof(sin);
   //接收客户端
   fd = accept(listener, (struct sockaddr *)&sin, &slen);
   if (fd < 0)
   {
     perror("error accept");
     return;
   }
   printf("ACCEPT: fd = %u\n", fd);
   ////注册一个bufferevent_socket_new事件
   struct bufferevent *bev = bufferevent_socket_new(base, fd, BEV_OPT_CLOSE_ON_FREE);
   ////设置回掉函数
   bufferevent_setcb(bev, read_cb, NULL, error_cb, arg);
   ////设置该事件的属性
   bufferevent_enable(bev, EV_READ | EV_WRITE | EV_PERSIST);
 }
 ////read 回调函数
 void read_cb(struct bufferevent *bev, void *arg)
 {
 #define MAX_LINE 256
   char line[MAX_LINE + 1];
   int n;
   //通过传入参数bev找到socket fd
   evutil_socket_t fd = bufferevent_getfd(bev);
   //
   while (n = bufferevent_read(bev, line, MAX_LINE))
   {
     line[n] = '\0';
     printf("fd=%u, read line: %s\n", fd, line);
     //将获取的数据返回给客户端
     bufferevent_write(bev, line, n);
   }
 }
 ////error回调函数
 void error_cb(struct bufferevent *bev, short event, void *arg)
 {
   //通过传入参数bev找到socket fd
   evutil_socket_t fd = bufferevent_getfd(bev);
   //cout << "fd = " << fd << endl;
   if (event & BEV_EVENT_TIMEOUT)
   {
     printf("Timed out\n"); //if bufferevent_set_timeouts() called
   }
   else if (event & BEV_EVENT_EOF)
   {
     printf("connection closed\n");
   }
   else if (event & BEV_EVENT_ERROR)
   {
     printf("some other error\n");
   }
   bufferevent_free(bev);
 }
 ////write 回调函数
 void write_cb(struct bufferevent *bev, void *arg)
 {
   char str[50];
   //通过传入参数bev找到socket fd
   evutil_socket_t fd = bufferevent_getfd(bev);
   //cin >> str;
   printf("输入数据!");
   scanf_s("%d", &str);
   bufferevent_write(bev, &str, sizeof(str));
 }

 int main()
 {
   int ret;
   evutil_socket_t listener;
   WSADATA Ws;
   //Init Windows Socket
   if (WSAStartup(MAKEWORD(2, 2), &Ws) != 0)
   {
     return -1;
   }
   listener = socket(AF_INET, SOCK_STREAM, 0);
   assert(listener > 0);
   evutil_make_listen_socket_reuseable(listener);
   struct sockaddr_in sin;
   sin.sin_family = AF_INET;
   sin.sin_addr.s_addr = 0;
   sin.sin_port = htons(LISTEN_PORT);
   if (bind(listener, (struct sockaddr *)&sin, sizeof(sin)) < 0) {
     perror("bind");
     return 1;
   }
   if (listen(listener, 1000) < 0) {
     perror("listen");
     return 1;
   }
  printf("Listening...\n");
   evutil_make_socket_nonblocking(listener);
   struct event_base *base = event_base_new();
   assert(base != NULL);
   struct event *listen_event;
   listen_event = event_new(base, listener, EV_READ | EV_PERSIST, do_accept_cb, (void*)base);
   event_add(listen_event, NULL);
   event_base_dispatch(base);
   printf("The End.");
   return 0;
 }

客户端代码:Client.cpp

/******* 客户端程序 client.c ************/
#define _WINSOCK_DEPRECATED_NO_WARNINGS
#define _CRT_SECURE_NO_WARNINGS
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include<winsock2.h>
#include<ws2tcpip.h>
#include<iostream>

 #pragma comment (lib,"ws2_32.lib")
 int main(int argc, char *argv[])
 {
   WSADATA Ws;
   //Init Windows Socket
   if (WSAStartup(MAKEWORD(2, 2), &Ws) != 0)
   {
     return 0;
   }
   int sockfd;
   char buffer[1024];
   struct sockaddr_in server_addr;
   struct hostent *host;
   int portnumber, nbytes;

   if ((host = gethostbyname("127.0.0.1")) == NULL)
   {
     fprintf(stderr, "Gethostname error\n");
     exit(1);
   }

   if ((portnumber = atoi("9999"))<0)
   {
     fprintf(stderr, "Usage:%s hostname portnumber\a\n", argv[0]);
     exit(1);
   }

   /* 客户程序开始建立 sockfd描述符 */
   if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1)
   {
     fprintf(stderr, "Socket Error:%s\a\n", strerror(errno));
     exit(1);
   }

   /* 客户程序填充服务端的资料    */
   memset(&server_addr,0, sizeof(server_addr));
   server_addr.sin_family = AF_INET;
   server_addr.sin_port = htons(portnumber);
   server_addr.sin_addr = *((struct in_addr *)host->h_addr);

   /* 客户程序发起连接请求     */
   if (connect(sockfd, (struct sockaddr *)(&server_addr), sizeof(struct sockaddr)) == -1)
  {
     fprintf(stderr, "Connect Error:%s\a\n", strerror(errno));
     exit(1);
   }

   while (true)
   {
     char MESSAGE[] = "hello server..\n";
     //bufferevent_write(buf_ev,MESSAGE,strlen(MESSAGE));
     //
     if (-1 == (::send(sockfd, MESSAGE, strlen(MESSAGE), 0)))
     {
       printf("the net has a error occured..");
       break;
     }

     if ((nbytes = recv(sockfd, buffer, 1024,0)) == -1)
    {
       fprintf(stderr, "read error:%s\n", strerror(errno));
       exit(1);
     }

     buffer[nbytes] = '\0';
     printf("I have received:%s\n", buffer);
     memset(buffer, 0, 1024);

     Sleep(2);

   }
   /* 结束通讯   */
   closesocket(sockfd);
   exit(0);

   return 0;
 }

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索windows
, 服务器
libevent
libevent服务器、libevent 服务器例子、libevent http服务器、libevent tcp服务器、libevent 游戏服务器,以便于您获取更多的相关知识。

时间: 2024-11-13 08:50:45

Windows下实现简单的libevent服务器_服务器其它的相关文章

Windows下Apache+Tomcat+MySQL+jsp+php的服务器整合配置经验总结

apache|js|mysql|window|服务器 Windows下Apache+Tomcat+MySQL+jsp+php的服务器整合配置经验总结 1.作者:moonsbird 题目:Windows下Apache+Tomcat+MySQL+jsp+php的服务器整合配置经验总结 时间:2004.8.19 最初发表于西南交通大学锦城驿站 2.本文是作者学习几年来学习jsp/配置服务器环境的经验总结.可以转载,但请注明出处. 开始学习jsp很久了,网上有许多关于jsp整合的例子,但存在着许多问题.

Windows下Apache应用环境塔建方法_win服务器

目的:为Apache,php配置受限制的用户权限 环境配置情况:apache安装目录:d:\www-s\apachephp目录:d:\www-s\php5mysql目录:d:\www-s\mysql网站根目录:d:\www\htdocs 专门为运行Apache运行所使用的用户:apache-u(可不隶属于任何用户组) PS:这里只说Windows下Apache应用环境相关的目录权限设置,至于其他基本的服务器目录权限设置就不提啦! Windows下Apache应用环境塔建目录安全设置操作步骤: 配

Windows下用MyEclipse作开发链接Linux服务器的数据库

问题描述 本人最近学习Linux系统,在windows虚拟机下的Linux系统中安装了tomcate,搭建J2ee开发环境!通过windows下的eclipse连接虚拟机中的tomate服务器,做一些J2ee项目,Linux下的数据库是oracle9i,目前没有什么头绪,希望各位高手多多指点!介绍一些成功的方法!谢谢!!! 解决方案 解决方案二:我也在研究,还没搞清楚,沙发一个,等专家的答复.解决方案三:等待专家!

使用Apache&amp;花生壳架设Web服务器_服务器

这里讲的是IIS服务器,其实花生壳在Apache下也能正常使用,这一点我在以前的帖子中已经讲过,现在专门把Apache的配置作一详细说明: 首先:当然是下载Apache啦,Apache目前最新的Windows版本是1.3.2X,你可以到它的官方网站去下载最新版(http://www.apache.org),Win2000对应的下载文件是Win32版本,Win98和Winme对应的下载文件是R298orMe版本,注意别下载错了.如果你使用的是Win2000而下载了Win98orMe的版本,嘿嘿,等

windows 2003配置IIS支持.shtml .shtm 的简单方法 图文教程_服务器

今天帮朋友开一空间,他程序是shtml的,空间开好了,但是朋友说不支持,结果找了一下原因.其实windows 2003安装好IIS之后默认是支持.shtml的,只要在"WEB服务扩展"允许"在服务器前端的包含文件"即可,如下图:480) {this.resized=true; this.width=480;}" border=0 resized="true">

红客必学:Windows下的权限设置详解_网络冲浪

随着动网论坛的广泛应用和动网上传漏洞的被发现以及SQL注入式攻击越来越多的被使用,WEBSHELL让防火墙形同虚设,一台即使打了所有微软补丁.只让80端口对外开放的WEB服务器也逃不过被黑的命运.难道我们真的无能为力了吗?其实,只要你弄明白了NTFS系统下的权限设置问题,我们可以对crackers们说:NO! 要打造一台安全的WEB服务器,那么这台服务器就一定要使用NTFS和Windows NT/2000/2003.众所周知,Windows是一个支持多用户.多任务的操作系统,这是权限设置的基础,

在Windows下自动备份PostgreSQL的教程_数据库其它

背景在我工作上一个使用PostgreSQL数据库的项目上需要一个自动化系统来每天执行备份.经过一番研究决定通过创建一个Windows批处理文件并添加到Windows计划任务中来实现. 下面是具体步骤: 怎样配置第一步: 下载批处理文件. 第二步: 你可以通过一个简单的命令(schtasks /?查看帮助)或者使用图形界面(开始-控制面板-系统和安全-管理工具-任务计划程序)运行任务计划管理工具,还可以在%SYSTEMROOT%\System32目录下双击Taskschd.msc来启动它.   第

Windows 下 MySQL 简单定时自动备份、删除过期备份

问题 MySQL Workbench 客户端虽然好用,但并不提供自动备份功能.手工备份,确实繁琐. 环境 Windows Server 2012 MySQL 5.6.24 思考 MySQL 提供了 mysqldump 来进行备份.那么我们可否使用该工具,结合Windows 的定时任务功能,来实现 MySQL 定时自动备份呢? 解决 新建一个 数据库备份文件存放目录,本例为D:\db_backup.新建一个批处理文件,可以起任意名,本例为mysql_backup_tool.bat ,文件内容如下:

windows下编译及使用libevent

Libevent官网:http://libevent.org/ windows 7下编译: 编译环境: windows 7 + VS2010 (1)解压libevent到F:\libevent\libevent-2.0.21-stable (2)打开Microsoft visual studio 2010命令行工具 (3)修改以下三个文件,添加宏定义: 在以下3个文件开头添加"#define _WIN32_WINNT 0x0500" libevent-2.0.21-stable\eve