栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > C/C++/C#

libevent简单服务端和客户端实现

C/C++/C# 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

libevent简单服务端和客户端实现

1.生成动态库说明

在开始使用之前,我们需要先搞清楚libevent编译生成的各个动态库的作用。
在下载libevent源码包进行编译以后,当前目录生成.libs目录,该目录下是所有的目标文件,这里我们只说明个动态库so文件的作用,通过makefile我们可以知道各个动态库包含内容:

动态库名称 作用
libevent_core.so 这个库包含了所有核心的事件和缓存功能
libevent_extra.so 这个库包含外围的dns、rpc、http等协议使用
libevent.so 这个库包含了libevent_core和libevent_extra的内容
libevent_openssl.so 需要进行加密通信时可以使用这个
libevent_pthreads.so 看名字就知道如果要用多线程的方式使用libevent,就需要用到这个库

下面使用libevent实现了一个很简单的服务端和客户端程序。

2. 服务端代码实现
//server-event.cpp
#include 
#include 
#include 
#include 
#include 
#include 
#include 

#include 
#include 
#include 
#include 
#include 

static const int PORT = 9995;

static char g_szWriteMsg[256] = {0};
static char g_szReadMsg[256] = {0};
static int g_iCnt = 0;

static void listener_cb(struct evconnlistener *, evutil_socket_t,
    struct sockaddr *, int socklen, void *);
static void conn_writecb(struct bufferevent *, void *);
static void conn_readcb(struct bufferevent *, void *);
static void conn_eventcb(struct bufferevent *, short, void *);

int
main(int argc, char **argv)
{
	struct event_base *base;
	struct evconnlistener *listener;
	struct event *signal_event;

	struct sockaddr_in sin;

	base = event_base_new();
	if (!base) {
		fprintf(stderr, "Could not initialize libevent!n");
		return 1;
	}

	memset(&sin, 0, sizeof(sin));
	sin.sin_family = AF_INET;
	sin.sin_port = htons(PORT);//固定一个端口号

	//创建、绑定、监听socket
	listener = evconnlistener_new_bind(base, listener_cb, (void *)base,
	    LEV_OPT_REUSEABLE|LEV_OPT_CLOSE_ON_FREE, -1,
	    (struct sockaddr*)&sin,
	    sizeof(sin));

	if (!listener) {
		fprintf(stderr, "Could not create a listener!n");
		return 1;
	}	

	event_base_dispatch(base);

	evconnlistener_free(listener);
	//event_free(signal_event);
	event_base_free(base);

	printf("donen");
	return 0;
}

//有连接来时调用
static void
listener_cb(struct evconnlistener *listener, evutil_socket_t fd,
    struct sockaddr *sa, int socklen, void *user_data)
{
	struct event_base *base = (struct event_base*)user_data;
	struct bufferevent *bev;

	//构造一个bufferevent
	bev = bufferevent_socket_new(base, fd, BEV_OPT_CLOSE_ON_FREE);
	if (!bev) {
		fprintf(stderr, "Error constructing bufferevent!");
		event_base_loopbreak(base);
		return;
	}

	//绑定读事件回调函数、写事件回调函数、错误事件回调函数
	bufferevent_setcb(bev, conn_readcb, conn_writecb, conn_eventcb, NULL);

	bufferevent_enable(bev, EV_WRITE);
	bufferevent_enable(bev, EV_READ);

	const char *szMsg = "hi client!";
	bufferevent_write(bev, szMsg, strlen(szMsg));
}

static void
conn_writecb(struct bufferevent *bev, void *user_data)
{
	//printf("touch conn_writecbn");
	
//	if ( strlen(g_szWriteMsg) > 0 )
//	{
//		bufferevent_write(bev, g_szWriteMsg, strlen(g_szWriteMsg));
//		memset(g_szWriteMsg, 0x00, sizeof(g_szWriteMsg));
//	}
}

static void
conn_readcb(struct bufferevent *bev, void *user_data)
{
	//printf("touch conn_readcbn");
	memset(g_szReadMsg, 0x00, sizeof(g_szReadMsg));
	struct evbuffer *input = bufferevent_get_input(bev);
	size_t sz = evbuffer_get_length(input);
	if (sz > 0)
	{
		bufferevent_read(bev, g_szReadMsg, sz);
		printf("cli:>>%sn", g_szReadMsg);
		memset(g_szWriteMsg, 0x00, sizeof(g_szWriteMsg));
		snprintf(g_szWriteMsg, sizeof(g_szWriteMsg)-1, "hi client, this count is %d", g_iCnt);
		g_iCnt++;
		//printf("ser:>>");
		//gets(g_szWriteMsg);
		//scanf("%s", g_szWriteMsg);
		
		bufferevent_write(bev, g_szWriteMsg, strlen(g_szWriteMsg));
	}
}

static void
conn_eventcb(struct bufferevent *bev, short events, void *user_data)
{
	if (events & BEV_EVENT_EOF) {
		printf("Connection closed.n");
	} else if (events & BEV_EVENT_ERROR) {
		printf("Got an error on the connection: %sn",
		    strerror(errno));
	}
	
	bufferevent_free(bev);
}
3. 客户端代码实现
//client-event.cpp
#include 
#include 
#include 
#include 
#include 
#include 
#include 

#include 
#include 
#include 
#include 
#include 

static const int PORT = 9995;
static char g_szWriteMsg[256] = {0};
static char g_szReadMsg[256] = {0};
static int g_iCnt = 0;
static void conn_writecb(struct bufferevent *, void *);
static void conn_readcb(struct bufferevent *, void *);
static void conn_eventcb(struct bufferevent *, short, void *);

int
main(int argc, char **argv)
{
	struct event_base *base;

	struct sockaddr_in sin;

	base = event_base_new();
	if (!base) {
		fprintf(stderr, "Could not initialize libevent!n");
		return 1;
	}

	memset(&sin, 0, sizeof(sin));
	sin.sin_addr.s_addr = inet_addr("192.168.233.250");
	sin.sin_family = AF_INET;
	sin.sin_port = htons(PORT);
	
	struct bufferevent* bev = bufferevent_socket_new(base, -1, BEV_OPT_CLOSE_ON_FREE);
	if (bev == NULL )
	{
		fprintf(stderr, "socket init failedn");
		return 1;
	}
	bufferevent_setcb(bev, conn_readcb, conn_writecb, conn_eventcb, NULL);

	//连接服务端
	int flag = bufferevent_socket_connect(bev, (struct sockaddr*)&sin, sizeof(sin));
	if (-1 == flag )
	{
		fprintf(stderr, "connect failedn");
		return 1;
	}
	bufferevent_enable(bev, EV_READ | EV_WRITE);

	event_base_dispatch(base);
	event_base_free(base);

	printf("donen");
	return 0;
}

static void
conn_writecb(struct bufferevent *bev, void *user_data)
{
	//printf("touch conn_writecbn");
	
//	if ( strlen(g_szWriteMsg) > 0 )
//	{
//		bufferevent_write(bev, g_szWriteMsg, strlen(g_szWriteMsg));
//		memset(g_szWriteMsg, 0x00, sizeof(g_szWriteMsg));
//	}
}

static void
conn_readcb(struct bufferevent *bev, void *user_data)
{
	//printf("touch conn_readcbn");
	memset(g_szReadMsg, 0x00, sizeof(g_szReadMsg));
	struct evbuffer *input = bufferevent_get_input(bev);
	size_t sz = evbuffer_get_length(input);
	if (sz > 0)
	{
		bufferevent_read(bev, g_szReadMsg, sz);
		printf("ser:>>%sn", g_szReadMsg);
		memset(g_szWriteMsg, 0, sizeof(g_szWriteMsg));
		snprintf(g_szWriteMsg, sizeof(g_szWriteMsg)-1, "hi server,this count is %d", g_iCnt);
		g_iCnt++;
		//printf("cli:>>");
		//gets(g_szWriteMsg);
		//scanf("%s", g_szWriteMsg);
		bufferevent_write(bev, g_szWriteMsg, strlen(g_szWriteMsg));
	}
}

static void
conn_eventcb(struct bufferevent *bev, short events, void *user_data)
{
	if (events & BEV_EVENT_EOF) {
		printf("Connection closed.n");
	} else if (events & BEV_EVENT_ERROR) {
		printf("Got an error on the connection: %sn",
		    strerror(errno));
	}
	else if(events & BEV_EVENT_CONNECTED)
	{
		//连接成功时走这里,并且要客户端第一次触发读事件后连接才真正建立起来
		printf("connect successn");
		const char* msg = "hi server,hao are you";
		bufferevent_write(bev, msg, strlen(msg));
		return;
	}
	bufferevent_free(bev);
}

上面服务端和客户端代码使用libevent建立了一个简单的聊天应用,编译时需链接-levent_core。

文章同步发表在cpp加油站(ID:xy13640954449), 欢迎关注!

转载请注明:文章转载自 www.mshxw.com
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号