libevent::实现的低级 socket
LibEvent 实现的低级 ROT13 例子
#include <cstdio>
#include <netinet/in.h>
#include <sys/socket.h>
#include <fcntl.h>
#include <event2/event.h>
#include <assert.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <errno.h> #define MAX_LINE 16384 struct fd_state
{
char buffer[MAX_LINE];
size_t buffer_used;
size_t n_written;
size_t write_upto;
struct event*read_event;
struct event*write_event;
}; void run(void);
void do_accept(evutil_socket_t listener, short event, void*arg);
struct fd_state* alloc_fd_state(struct event_base*base, evutil_socket_t fd);
void do_read(evutil_socket_t fd, short events, void*arg);
void do_write(evutil_socket_t fd, short events, void*arg);
void free_fd_state(struct fd_state*state); char rot13_char(char c)
{
if ((c >= 'a' && c <= 'm ') || (c >= 'A ' && c <= 'M '))
return c + ;
else if ((c >= 'n' && c <= 'z') || (c >= 'N' && c <= 'Z'))
return c - ;
else
return c;
} int main(int c, char**v)
{
run();
return ;
} void run(void)
{
evutil_socket_t listener;
struct sockaddr_in sin;
struct event_base*base;
struct event*listener_event;
base = event_base_new();
if (!base)
return; sin.sin_family = AF_INET; sin.sin_addr.s_addr = ;
sin.sin_port = htons();
listener = socket(AF_INET, SOCK_STREAM, );
evutil_make_socket_nonblocking(listener);
/*#ifndef WIN32
{
int one = 1;
setsockopt(listener, SOL_SOCKET, SO_REUSEADDR, &one, sizeof(one));
}
#endif */
if (bind(listener, (struct sockaddr*) &sin, sizeof(sin)) < )
{
perror("bind"); return;
}
if (listen(listener, ) < )
{
perror("listen");
return;
}
listener_event = event_new(base, listener, EV_READ | EV_PERSIST, do_accept, (void*)base); event_add(listener_event, NULL);
event_base_dispatch(base);
} void do_accept(evutil_socket_t listener, short event, void*arg)
{
struct event_base*base = (event_base*)(arg);
struct sockaddr_storage ss;
socklen_t slen = sizeof(ss);
int fd = accept(listener, (struct sockaddr*) &ss, &slen);
if (fd < )
{
perror("accept");
}
else if (fd > FD_SETSIZE)
{
close(fd);
}
else
{
struct fd_state* state; evutil_make_socket_nonblocking(fd); state = alloc_fd_state(base, fd);
assert(state);
assert(state->write_event);
event_add(state->read_event, NULL);
}
} struct fd_state* alloc_fd_state(struct event_base*base, evutil_socket_t fd)
{
fd_state* state = (fd_state*)malloc(sizeof(fd_state));
if (!state) return NULL;
state->read_event = event_new(base, fd, EV_READ | EV_PERSIST, do_read, state);
if (!state->read_event)
{
free(state); return NULL;
}
state->write_event = event_new(base, fd, EV_WRITE | EV_PERSIST, do_write, state);
if (!state->write_event)
{
event_free(state->read_event);
free(state);
return NULL;
}
state->buffer_used = state->n_written = state->write_upto = ;
assert(state->write_event);
return state;
} void do_read(evutil_socket_t fd, short events, void*arg)
{
struct fd_state*state = (fd_state*)arg;
int i;
ssize_t result;
while ()
{
char buf[] = { '\0' };
assert(state->write_event);
result = recv(fd, buf, sizeof(buf), );
if (result <= )
break; printf("buf = [ %s ]\n", buf); assert(state->write_event);
event_add(state->write_event, NULL);
state->write_upto = state->buffer_used; for (i = ; i < result; ++i)
{
if (state->buffer_used < sizeof(state->buffer))
state->buffer[state->buffer_used++] = rot13_char(buf[i]); if (buf[i] == '\n')
{
assert(state->write_event);
event_add(state->write_event, NULL);
state->write_upto = state->buffer_used;
}
}
}
if (result == )
{
free_fd_state(state);
}
else if (result < )
{
if (errno == EAGAIN)
return; perror("recv");
free_fd_state(state);
}
} void do_write(evutil_socket_t fd, short events, void*arg)
{
struct fd_state*state = (fd_state*)arg;
while (state->n_written < state->write_upto)
{
ssize_t result = send(fd, state->buffer + state->n_written, state->write_upto - state->n_written, );
if (result < )
{
if (errno == EAGAIN)
return; free_fd_state(state);
return;
}
assert(result != );
state->n_written += result;
}
if (state->n_written == state->buffer_used)
{
state->n_written = state->write_upto = state->buffer_used = ;
}
event_del(state->write_event);
} void free_fd_state(struct fd_state*state)
{
event_free(state->read_event);
event_free(state->write_event);
free(state);
}
libevent::实现的低级 socket的更多相关文章
- libevent (一) socket属性设置与初始化操作
socket属性设置与初始化操作 libevent是一个事件触发的网络库,适用于windows.linux.bsd等多种平台,内部使用select.epoll.kqueue等系统调用管理事件机制.著名 ...
- 基于libevent和unix domain socket的本地server
https://www.pacificsimplicity.ca/blog/libevent-echo-server-tutorial 根据这一篇写一个最简单的demo.然后开始写client. cl ...
- 使用libevent进行多线程socket编程demo
最近要对一个用libevent写的C/C++项目进行修改,要改成多线程的,故做了一些学习和研究. libevent是一个用C语言写的开源的一个库.它对socket编程里的epoll/select等功能 ...
- 网络(一),libevent客户端部分
网络模块() 一.服务端: 暂时就以libevent模块,共享内存等下 .GS打开,首先创建4个libevent子线程,当然为每个线程设置连接通知回调函数,这个是基于sockpair的,然后再创建一个 ...
- C10K问题和Libevent库介绍
http://blog.chinaunix.net/uid-20761674-id-75056.html 一.C10K的问题 C10K的问题在上个世纪90年代就被提出来了.大概的意思是当用户数超过1万 ...
- 【Socket】linux网络多路复用IO技术
1.mystery引入 1)Select是一种多路复用IO输入输出模式,在linux的输入输出编程中通过select的轮询机制,发现可用/可读或可写的接口. 2)低级socket程 ...
- libevent源码剖析
libevent是一个使用C语言编写的,轻量级的开源高性能网络库,使用者很多,研究者也很多.由于代码简洁,设计思想简明巧妙,因此很适合用来学习,提升自己C语言的能力. libevent有这样显著地几个 ...
- libevent编程疑难解答
http://blog.csdn.net/luotuo44/article/details/39547391 转载请注明出处:http://blog.csdn.net/luotuo44/article ...
- libevent reference Mannual V -- Bufferevents
FYI: http://www.wangafu.net/~nickm/libevent-book/Ref6_bufferevent.html Bufferevents: concepts and ba ...
随机推荐
- 模板引擎Velocity学习系列-#set指令
#set指令 #set指令用于向一个变量或者对象赋值. 格式: #set($var = value) LHS是一个变量,不要使用特殊字符例如英文句号等,不能用大括号括起来.测试发现#set($user ...
- 04 (H5*) Vue第四天
目录: 1:父组件向子组件传值,通过属性绑定的方式. 2:父组件向子组件传方法,通过事件绑定的方式 . 3:通过ref来获取Dom元素 1:父组件向子组件传值,通过属性绑定的方式 1.1:父组件声明数 ...
- 【PyTorch】Tricks 集锦
声明:本文大部分内容是从知乎.博客等知识分享站点摘录而来,以方便查阅学习.具体摘录地址已在文章底部引用部分给出. 1. 查看模型每层输出详情 from torchsummary import summ ...
- [LeetCode] 由 “分形" 所想
分形思想和递归思想有区别么? 一.简单例子 函数调用自己,简化了理解逻辑,但其他到处都是问题. #%% def listsum(numList): if len(numList) == 1: retu ...
- DefaultSerializer requires a Serializable payload but received an object of type [model.Admin]
一.问题描述: 在用redis做二级缓存时,出现如下异常 DefaultSerializer requires a Serializable payload but received an o ...
- 【面试必备】小伙伴栽在了JVM的内存分配策略。。。
周末有小伙伴留言说上周面试时被问到内存分配策略的问题,但回答的不够理想,小伙伴说之前公号里看过这一块的文章的,当时看时很清楚,也知道各个策略是干嘛的,但面试时脑子里清楚,心里很明白,但嘴里就是说不清楚 ...
- java架构之路-(分布式zookeeper)zookeeper集群配置和选举机制详解
上次博客我们说了一下zookeeper的配置文件,以及命令的使用https://www.cnblogs.com/cxiaocai/p/11597465.html.我们这次来说一下我们的zookeepe ...
- Spring 梳理 - @Component
使用@Component注解,表示该类定义为Spring管理Bean,使用默认value(可选)属性表示Bean标识符.如果不指定标识符,默认为首字母小写类名.例如类UserController的标识 ...
- Spring Boot 梳理 - 4个核心
Spring Boot 魔法的核心:自动配置.起步依赖.命令行界面.Actuator 自动配置: 不用手动配置JdbcTemplate的Bean 不用手动配置DataSource的Bean Sprin ...
- 小鸟初学Shell编程(七)变量引用及作用范围
变量引用 那么定义好变量,如何打印变量的值呢?举例下变量引用的方式. ${变量名}称作为对变量的引用 echo ${变量名}查看变量的值 ${变量名}在部分情况下可以省略成 $变量名 [root@li ...