网络开源框架之libev使用实例

本文演示了如何使用libev库创建一个简单的TCP服务器。通过socket初始化、设置监听、接收连接、读写回调函数,展示了libev在处理网络事件中的应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

// libev.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"


#include <stdio.h>
#include <sys/epoll.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <sys/types.h>
#include <signal.h>
#include <unistd.h>
#include <fcntl.h>
#include <string.h>
#include <errno.h>
#include <stdlib.h>
#include <pthread.h>
#include <ev.h>
#define MAXLEN 1023
#define PORT 8003
#define ADDR_IP "127.0.0.1"


int socket_init();
void accept_callback(struct ev_loop *loop, ev_io *w, int revents);
void recv_callback(struct ev_loop *loop, ev_io *w, int revents);
void write_callback(struct ev_loop *loop, ev_io *w, int revents);


int main(int argc, char** argv)
{
	int listen;
	ev_io ev_io_watcher;
	listen = socket_init();
	struct ev_loop *loop = ev_loop_new(EVBACKEND_EPOLL);
	ev_io_init(&ev_io_watcher, accept_callback, listen, EV_READ);
	ev_io_start(loop, &ev_io_watcher);
	ev_loop(loop, 0);
	ev_loop_destroy(loop);
	return 0;

}

int socket_init()
{
	struct sockaddr_in my_addr;
	int listener;
	if ((listener = socket(AF_INET, SOCK_STREAM, 0)) == -1)
	{
		perror("socket");
		exit(1);
	}
	else
	{
		printf("SOCKET CREATE SUCCESS!\n");
	}
	//setnonblocking(listener);
	int so_reuseaddr = 1;
	setsockopt(listener, SOL_SOCKET, SO_REUSEADDR, &so_reuseaddr, sizeof(so_reuseaddr));
	bzero(&my_addr, sizeof(my_addr));
	my_addr.sin_family = PF_INET;
	my_addr.sin_port = htons(PORT);
	my_addr.sin_addr.s_addr = inet_addr(ADDR_IP);

	if (bind(listener, (struct sockaddr *) &my_addr, sizeof(struct sockaddr)) == -1)
	{
		perror("bind error!\n");
		exit(1);
	}
	else
	{
		printf("IP BIND SUCCESS,IP:%s\n", ADDR_IP);
	}

	if (listen(listener, 1024) == -1)
	{
		perror("listen error!\n");
		exit(1);
	}
	else
	{
		printf("LISTEN SUCCESS,PORT:%d\n", PORT);
	}
	return listener;
}

void accept_callback(struct ev_loop *loop, ev_io *w, int revents)
{
	int newfd;
	struct sockaddr_in sin;
	socklen_t addrlen = sizeof(struct sockaddr);
	ev_io* accept_watcher = malloc(sizeof(ev_io));
	while ((newfd = accept(w->fd, (struct sockaddr *)&sin, &addrlen)) < 0)
	{
		if (errno == EAGAIN || errno == EWOULDBLOCK)
		{
			//these are transient, so don't log anything.
			continue;
		}
		else
		{
			printf("accept error.[%s]\n", strerror(errno));
			break;
		}
	}
	ev_io_init(accept_watcher, recv_callback, newfd, EV_READ);
	ev_io_start(loop, accept_watcher);
	printf("accept callback : fd :%d\n", accept_watcher->fd);

}

void recv_callback(struct ev_loop *loop, ev_io *w, int revents)
{
	char buffer[1024] = { 0 };
	int ret = 0;
	//ev_io write_event;
loop:
	ret = recv(w->fd, buffer, MAXLEN, 0);
	if (ret > 0)
	{
		printf("recv message :%s  \n", buffer);

	}
	else if (ret == 0)
	{
		printf("remote socket closed!socket fd: %d\n", w->fd);
		close(w->fd);
		ev_io_stop(loop, w);
		free(w);
		return;
	}
	else
	{
		if (errno == EAGAIN || errno == EWOULDBLOCK)
		{
			goto loop;
		}
		else
		{
			printf("ret :%d ,close socket fd : %d\n", ret, w->fd);
			close(w->fd);
			ev_io_stop(loop, w);
			free(w);
			return;
		}
	}
	int fd = w->fd;
	ev_io_stop(loop, w);
	ev_io_init(w, write_callback, fd, EV_WRITE);
	ev_io_start(loop, w);
	printf("socket fd : %d, turn read 2 write loop! ", fd);

}


void write_callback(struct ev_loop *loop, ev_io *w, int revents)
{
	char buffer[1024] = { 0 };
	//ev_io read_event;
	snprintf(buffer, 1023, "this is a libev server!\n");
	write(w->fd, buffer, strlen(buffer), 0);
	int fd = w->fd;
	ev_io_stop(loop, w);
	ev_io_init(w, recv_callback, fd, EV_READ);
	ev_io_start(loop, w);
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值