libevent高并发网络编程 - 02_libevent缓冲IO之bufferevent

news2025/1/13 7:22:23

文章目录

    • 1. 为什么需要缓冲区?
    • 2. 水位
    • 3. bufferevent常用API
      • 3.1 evconnlistener_new_bind()
      • 3.2 evconnlistener_free()
      • 3.3 bufferevent_socket_new()
      • 3.4 bufferevent_enable()
      • 3.5 bufferevent_set_timeouts()
      • 3.6 bufferevent_setcb()
      • 3.7 bufferevent_setwatermark()
      • 3.8 bufferevent_write()
      • 3.9 bufferevent_read()
      • 3.10 bufferevent_free()
      • 3.11 bufferevent_trigger()
      • 3.12 bufferevent_socket_connect()
      • 3.13 evutil_inet_pton()
    • 4 bufferevent服务器接例子
    • 5 bufferevent客户端接例子

1. 为什么需要缓冲区?

  • libevent提供了一些高级别的网络编程抽象,其中包括缓冲式I/O的接口。在网络编程中,缓冲区是用来保存从网络中读取或向网络中写入数据的临时存储空间因为在网络通信中,传输速度和数据处理速度往往不匹配,所以使用缓冲区可以帮助应用程序更好地控制数据流量,避免阻塞和死锁等问题

  • libevent的 bufferevent 接口就提供了对缓冲区操作的封装,使得应用程序可以更轻松地管理和操作缓冲区。具体而言,bufferevent 会自动管理底层文件描述符(socket)上的读写缓冲区,并提供事件驱动的回调机制来实现异步的缓冲区读写操作。此外,bufferevent 还支持多种协议栈,如 TCP、UDP 和 SSL 等,以适应不同的网络应用场景。

2. 水位

bufferevent 中的水位是指缓冲区中数据的数量达到或超过一定阈值时触发相应的事件

具体而言,bufferevent 会维护两个水位,一个是读取的低水位(read low watermark),另一个是写入的高水位(write high watermark)。当缓冲区中的数据量高于读取低水位时,bufferevent 会触发读取事件;而当缓冲区中的数据量达到或超过写入高水位时,bufferevent 会暂停读取并触发写入事件,直到缓冲区中的数据量降到一定程度。

  • 假设有一个 TCP 连接上的 bufferevent,读取低水位设置为 1024 字节。当接收到数据时,如果缓冲区中可读数据量达到或超过 1024 字节,则 bufferevent 会处于可读状态并触发读取事件;如果缓冲区中的可读数据量低于 1024 字节,则 bufferevent 不会触发读取事件,直到缓冲区中的可读数据量达到或超过 1024 字节时才开始触发读取事件。

  • 假设有一个 TCP 连接上的 bufferevent,写入高水位设置为 4096 字节。当发送数据时,如果缓冲区中的可写数据量达到或超过 4096 字节,则 bufferevent 会处于不可写状态并触发写入事件;如果缓冲区中的可写数据量低于 4096 字节,则 bufferevent 不会触发写入事件,直到缓冲区中的可写数据量达到或超过 4096 字节时才开始触发写入事件。

触发写入事件不意味着立即将缓冲区中的数据全部写入网络中。实际上,bufferevent 会尽可能地将缓冲区中的数据写入网络中,但也要考虑到网络带宽、延迟和拥塞等因素,以确保数据传输的效率和可靠性。

3. bufferevent常用API

3.1 evconnlistener_new_bind()

evconnlistener_new_bind() 函数用于创建监听套接字,并将其添加到 libevent 的事件处理器中,等待客户端连接。

具体来说,它可以用于实现以下功能:

  1. TCP 服务器:通过调用 evconnlistener_new_bind() 函数创建一个监听套接字,然后使用 listen() 函数开始监听客户端连接请求。当客户端连接到达时,libevent 将会自动触发相应的事件回调函数,可以在回调函数中进行新连接的处理和数据读写等操作。
  2. UDP 服务器:虽然 UDP 是无连接的协议,但同样可以使用 evconnlistener_new_bind() 函数创建一个本地套接字并绑定到指定地址和端口上,以接收远程主机发送的 UDP 数据包。当有数据包到达时,libevent 将会触发相应的事件回调函数,开发者可以在回调函数中进行数据包的处理和响应等操作。
  3. 其他网络应用:除了常见的 TCP/UDP 服务器外,evconnlistener_new_bind() 还可以用于创建各种类型的网络应用,例如 HTTP 服务器、WebSocket 服务器、DNS 服务器等。

总之,evconnlistener_new_bind() 函数是 libevent 库中实现高效异步网络编程所必需的重要接口之一,可以帮助更方便地创建、管理和处理网络连接。

函数原型:

struct evconnlistener *
evconnlistener_new_bind(struct event_base *base, evconnlistener_cb cb,
    void *ptr, unsigned flags, int backlog, 
    const struct sockaddr *sa, int socklen)
    
参数说明:
    base:事件处理器所使用的 event_base 对象;
    cb:新连接到来时调用的回调函数;
    ptr:传递给回调函数的参数指针;
    flags:监听套接字的标志位,可以使用 LEV_OPT_CLOSE_ON_FREE、LEV_OPT_REUSEABLE 和 LEV_OPT_THREADSAFE 进行设置;
    backlog:侦听套接字的挂起连接队列的最大长度;
    sa:sockaddr 结构体指针,表示要绑定的地址和端口号;
    socklen:地址结构体的长度。
    
返回evconnlistener对象
    
flags 参数用于设置监听套接字的属性,具体取值如下:
    LEV_OPT_CLOSE_ON_FREE:在 evconnlistener_free() 函数调用时自动关闭监听套接字,默认情况下不会关闭。
    LEV_OPT_REUSEABLE:设置监听套接字为可重用状态,即使在套接字被关闭后仍然可以重复使用同一个地址。
    LEV_OPT_THREADSAFE:开启线程安全模式,支持多个线程同时访问同一个 evconnlistener 对象。
例如,要同时设置可重用和自动关闭属性,可以将 flags 参数设置为 LEV_OPT_CLOSE_ON_FREE | LEV_OPT_REUSEABLE。需要注意的是,在使用多个 evconnlistener 对象进行端口监听时,如果想要复用同一个地址和端口号,必须要将所有的监听套接字都设置为可重用状态。
    
evconnlistener_cb函数指针是libevent网络库中的一个回调函数指针类型,用于指向监听器接受新连接时的回调函数。该回调函数需要有如下形式:
void evconnlistener_cb(struct evconnlistener *listener, 
                        evutil_socket_t fd, 
                        struct sockaddr *addr, 
                        int socklen, 
                        void *ctx);

evconnlistener_cb函数指针是evconnlistener_new_bind()回调函数指针类型,用于指向监听器接受新连接时的回调函数。该回调函数需要有如下形式:

void evconnlistener_cb(struct evconnlistener *listener, 
                        evutil_socket_t fd, 
                        struct sockaddr *addr, 
                        int socklen, 
                        void *ctx);
参数说明:
    listener:指向evconnlistener监听器对象的指针。
    fd:表示新收到连接的socket文件描述符。
    addr:指向客户端地址信息的结构体指针。
    socklen:客户端地址结构体的长度。
    ctx:监听器上下文参数,在创建监听器时传入。
在evconnlistener_cb回调函数中,可以对新收到的连接进行处理,例如创建一个bufferevent缓冲事件或者直接进行数据读写等操作。

调用 evconnlistener_new_bind() 函数会创建一个新的 evconnlistener 对象,并将其添加到事件处理器中。该函数会自动创建并绑定一个服务器套接字,等待客户端连接。当有新的连接请求到达后,libevent 将会调用 cb 函数并传递连接套接字描述符作为参数,同时 ptr 指向 evconnlistener_new_bind() 中传递的参数。

3.2 evconnlistener_free()

evconnlistener_free() 函数用于释放一个已经创建的evconnlistener监听器对象,释放该对象所占用的内存空间。

void evconnlistener_free(struct evconnlistener *lev)

3.3 bufferevent_socket_new()

bufferevent_socket_new()用于创建一个基于socket的bufferevent缓冲事件对象的函数。将该缓冲事件对象与指定的socket文件描述符关联通过该函数创建的缓冲事件对象来进行数据的读写操作,并且自动管理内部缓存区数据的处理

  • 该函数还支持设置多种缓冲策略和水位参数,以适应不同业务需求。例如,可以设置缓冲区大小、读写水位线等参数,来实现流控制和提高网络传输的效率。

  • bufferevent_socket_new()函数还能够设置回调函数,处理缓冲区数据读写事件或者错误事件等

函数的原型如下:

struct bufferevent *
bufferevent_socket_new(struct event_base *base, evutil_socket_t fd, int options)
    
参数说明:
    base:指向event_base事件处理器对象的指针。
    fd:表示要关联到bufferevent的已经打开的socket文件描述符。
    options:表示缓冲事件的配置选项。
    
返回值bufferevent对象
    
options参数来设置缓冲事件的配置选项。常用的一些选项:
    BEV_OPT_CLOSE_ON_FREE:释放缓冲事件时自动关闭关联的socket文件描述符。
    BEV_OPT_THREADSAFE:将缓冲事件对象设为线程安全模式。
    BEV_OPT_DEFER_CALLBACKS:延迟回调机制,可以提高多个事件的处理效率。
    BEV_OPT_UNLOCK_CALLBACKS:解锁回调机制,可以支持多线程同时运行回调函数。
    BEV_EVENT_TIMEOUT
    BEV_EVENT_READING

3.4 bufferevent_enable()

bufferevent_enable()用于启用和关闭缓冲事件对象。将缓冲事件对象中开启对应的读、写模式,以便在有数据可读或可写时能够及时响应。

函数的原型如下:

int bufferevent_enable(struct bufferevent *bufev, short event)
    
参数说明:
    bufev:指向要启用的bufferevent缓冲事件对象的指针。
    event:表示要启用的事件类型,可以是EV_READ、EV_WRITE或者EV_EVENT组合。 
    
常用的event类型包括:
    EV_TIMEOUT:超时事件,用于实现定时器功能。
    EV_READ:读事件,表示该事件关心要读取的文件描述符是否可读。
    EV_WRITE:写事件,表示该事件关心要写入的文件描述符是否可写。
    EV_SIGNAL:信号事件,表示该事件与系统信号相关联。
    EV_PERSIST:持久性事件,表示注册的事件触发后不会自动被删除,而是需要手动删除。
    EV_ET:边缘触发事件,表示只有当事件状态发生变化时才通知监听者。
    EV_FINALIZE:最终化事件,表示事件处理器即将销毁之前,在内存释放时调用一次。
    EV_CLOSED:连接关闭事件,用于TCP连接管理。

bufferevent_enable()函数也可以关闭指定的缓冲事件对象,只需将event参数设置为0即可,例如:

bufferevent_enable(bufev, 0); // 关闭缓冲事件对象

3.5 bufferevent_set_timeouts()

bufferevent_set_timeouts()函数的作用是设置指定bufferevent缓冲事件对象的读写超时时间。该函数可以让在网络传输过程中,对数据读取或写入的响应时间进行限制,从而实现流控制和避免网络拥塞等问题。

该函数的原型如下:

int
bufferevent_set_timeouts(struct bufferevent *bufev,
			 const struct timeval *tv_read,
			 const struct timeval *tv_write)
    
参数说明:
    bufev:指向要设置超时时间的bufferevent缓冲事件对象的指针。
    timeout_read:表示读超时时间的timeval结构体指针,如果为NULL,则不设置读超时时间。
    timeout_write:表示写超时时间的timeval结构体指针,如果为NULL,则不设置写超时时间。

3.6 bufferevent_setcb()

bufferevent_setcb()函数的作用是设置指定bufferevent缓冲事件对象的回调函数。该函数可以让在缓冲事件发生读写或者错误等事件时,及时进行处理和响应。

该函数的原型如下:

void
bufferevent_setcb(struct bufferevent *bufev,
    bufferevent_data_cb readcb, bufferevent_data_cb writecb,
    bufferevent_event_cb eventcb, void *cbarg)
    
参数说明:
    bufev:指向要设置回调函数的bufferevent缓冲事件对象的指针。
    readcb:表示读取数据时的回调函数指针,当有数据可读时会调用该函数。
    writecb:表示写入数据时的回调函数指针,当缓冲区可写时会调用该函数。
    eventcb:表示异常事件(例如错误、连接关闭等)时的回调函数指针,当异常事件发生时会调用该函数。
    cbarg:表示回调函数的上下文参数,在调用回调函数时会传递给它。

readcbwritecbeventcb这三个回调函数形式,如下所示:

void readcb(struct bufferevent *bev, void *ctx) {
    /* 处理读取缓冲区中的数据 */
}

void writecb(struct bufferevent *bev, void *ctx) {
    /* 处理写入缓冲区的数据 */
}

void eventcb(struct bufferevent *bev, short what, void *ctx) {
    if (what & BEV_EVENT_ERROR) {
        /* 处理错误事件 */
    }
    if (what & BEV_EVENT_TIMEOUT) {
        /* 处理超时事件 */
    }
    if (what & BEV_EVENT_EOF) {
        /* 处理连接关闭事件 */
    }
}

bufferevent_setcb(bufev, readcb, writecb, eventcb, NULL);

其中,bev参数表示发生事件的缓冲事件对象本身,ctx参数表示回调函数的上下文参数。

3.7 bufferevent_setwatermark()

bufferevent_setwatermark()函数的作用是设置指定bufferevent缓冲事件对象的读写水位线。该函数可以让网络传输过程中,控制缓冲区数据的读取和写入,以实现流控制和避免网络拥塞等问题。

该函数的原型如下:

void bufferevent_setwatermark(struct bufferevent *bufev,
                              short events,
                              size_t lowmark,
                              size_t highmark);
参数说明:
    bufev:指向要设置水位线的bufferevent缓冲事件对象的指针。
    events:表示要设置水位线的事件类型,可以是BEV_READ、BEV_WRITE或者BEV_EVENT组合。
    lowmark:表示低水位线的大小,当缓冲区中数据量低于该值时会触发读取事件或可写事件。
    highmark:表示高水位线的大小,当缓冲区中数据量超过该值时会暂停读取事件或可写事件。

如果不想使用水位线机制,可以将lowmarkhighmark参数都设为0。另外,bufferevent_setwatermark()函数只对BEV_READBEV_WRITE事件有效,对其他事件类型无效。

3.8 bufferevent_write()

bufferevent_write()函数的作用是向指定的bufferevent缓冲事件对象中写入数据

该函数的原型如下:

int bufferevent_write(struct bufferevent *bufev, const void *data, size_t size)
    
数说明:
    bufev:指向要写入数据的bufferevent缓冲事件对象的指针。
    data:表示要写入数据的缓冲区指针。
    size:表示要写入数据的长度。

需要注意的是,想要保证写入数据的可靠性,可以在调用bufferevent_write()函数之前,先调用bufferevent_flush()函数将写入的数据从内存缓冲中刷到socket文件描述符中去。

如果希望对写入的数据进行批量处理,可以使用bufferevent_write_buffer()函数,它可以将一个EVBUFFER缓冲区中的数据全部或部分地写入到指定的bufferevent缓冲事件对象中去,例如:

struct evbuffer *output = bufferevent_get_output(bufev);
evbuffer_add(output, "Hello, world!", 13);
bufferevent_write_buffer(bufev, output);

3.9 bufferevent_read()

bufferevent_read()函数的作用是从指定的bufferevent缓冲事件对象中读取数据

该函数的原型如下:

size_t bufferevent_read(struct bufferevent *bufev, void *data, size_t size)

参数说明:
    bufev:指向要读取数据的bufferevent缓冲事件对象的指针。
    data:表示接收数据的缓冲区指针。
    size:表示要读取数据的长度。

如果希望对读取的数据进行批量处理,可以使用bufferevent_read_buffer()函数,它可以将一个EVBUFFER缓冲区中的数据全部或部分地读取到指定的bufferevent缓冲事件对象中去,例如:

struct evbuffer *input = bufferevent_get_input(bufev);
size_t len = evbuffer_get_length(input);
char *data = (char*) malloc(len);

if (bufferevent_read_buffer(bufev, input) == -1) {
    /* Handle error */
}

evbuffer_remove(input, data, len);
/* 处理读取到的数据 */
free(data);

3.10 bufferevent_free()

bufferevent_free()函数的作用是释放指定的bufferevent缓冲事件对象。

该函数的原型如下:

void bufferevent_free(struct bufferevent *bufev);
参数说明:
	bufev:指向要释放的bufferevent缓冲事件对象的指针。

使用bufferevent_free()函数释放bufferevent缓冲事件对象后,相关的定时器、EV_READ、EV_WRITE、EV_SIGNAL等事件也会被取消,并且与之关联的EVBUFFER对象也会被销毁。因此,在调用bufferevent_free()函数之前,需要确保已经处理完相关事件和缓冲区中的数据。

3.11 bufferevent_trigger()

bufferevent_trigger()函数的作用是触发指定bufferevent缓冲事件对象的读写事件或异常事件

该函数的原型如下:

void
bufferevent_trigger(struct bufferevent *bufev, short iotype, int options)
    
参数说明:
    bufev:指向要触发事件的bufferevent缓冲事件对象的指针。
    events:表示要触发的事件类型,可以是EV_READ、EV_WRITE或者EV_EVENT组合。
    options:可选项,为0表示仅触发一次事件,为BEV_TRIG_DEFERRED表示延迟触发事件。

需要注意的是,bufferevent_trigger()函数并不会真正地读取或写入数据,而只是模拟触发相应的事件,从而让事件回调函数被调用。

例如,以下代码可以手动触发一个读取事件:

bufferevent_trigger(bufev, BEV_EVENT_READ, 0);

3.12 bufferevent_socket_connect()

bufferevent_socket_connect()函数的作用是创建一个新的sockaddr结构体,并将其与指定的socket文件描述符绑定,然后通过该文件描述符连接到指定的服务器

该函数的原型如下:

int
bufferevent_socket_connect(struct bufferevent *bev,
    					   const struct sockaddr *sa, 
                           int socklen)
参数说明:
    bev:bufferevent缓冲事件对象的指针。
    sa:表示要连接的服务器地址的sockaddr结构体指针。
    socklen:表示服务器地址结构体的长度。

3.13 evutil_inet_pton()

evutil_inet_pton()函数的作用是将一个IPv4IPv6地址字符串转换为网络字节序格式的二进制地址。该函数们在网络编程中,方便地进行IP地址的转换和处理。

该函数的原型如下:

int evutil_inet_pton(int af, const char *src, void *dst);

参数说明:
    af:表示要转换的地址族类型,可以是AF_INET(IPv4)或AF_INET6(IPv6)。
    src:表示要转换的地址字符串。
    dst:指向存储转换结果的缓冲区指针。
    
返回值:如果转换成功,返回1;如果出现错误,返回0

evutil_inet_pton()函数会自动根据af参数选择相应的地址族类型,并将转换结果存储到dst指向的缓冲区中去。如果转换失败,返回0,并置errnoEINVAL等错误码。

例如,以下代码可以将一个IPv4地址字符串转换为网络字节序格式的二进制地址:

struct sockaddr_in sin;
memset(&sin, 0, sizeof(sin));

if (evutil_inet_pton(AF_INET, "127.0.0.1", &sin.sin_addr) <= 0) {
    /* Handle error */
}

4 bufferevent服务器接例子

#include <iostream>
#include <event2/event.h>
#include <event2/listener.h>
#include <event2/bufferevent.h>
#include <string.h>

#ifndef _WIN32
#include <signal.h>
#else
#endif

#include <errno.h>
#include <string.h>

using namespace std;

//错误,超时 (连接断开会进入)
void event_cb(bufferevent *be,short events,void *arg)
{
	cout<<"[E]"<<flush;	
	//读取超时时间发生后,数据读取停止
	if(events & BEV_EVENT_TIMEOUT && events & BEV_EVENT_READING)
	{
		cout<<"BEV_EVENT_READING BEV_EVENT_TIMEOUT"<<endl;
		//bufferevent_enable(be,EV_READ);
		bufferevent_free(be);
	}
	else if(events & BEV_EVENT_ERROR)
	{
		bufferevent_free(be);
	}
	else
	{
		cout<<"OTHERS"<<endl;
	}
}
void write_cb(bufferevent *be,void *arg)
{
	cout<<"[W]"<<flush;	
}
void read_cb(bufferevent *be,void *arg)
{
	cout<<"[R]"<<flush;
	char data[1024] = {0};
	//读取输入缓冲数据
	int len = bufferevent_read(be,data,sizeof(data)-1);
	cout<<"["<<data<<"]"<<endl;
	if(len<=0)return;
	if(strstr(data,"quit") !=NULL)
	{
		cout<<"quit";
		//退出并关闭socket BEV_OPT_CLOSE_ON_FREE
		bufferevent_free(be);
	}
	//发送数据 写入到输出缓冲
	bufferevent_write(be,"OK",3);
}


void listen_cb(evconnlistener *ev,evutil_socket_t s,sockaddr*sin,int slen,void *arg)
{
	cout<<"listen_cb"<<endl;
	event_base *base = (event_base *)arg;

	//创建bufferevent上下文 BEV_OPT_CLOSE_ON_FREE清理bufferevent时关闭socket
	bufferevent *bev = bufferevent_socket_new(base,	//指向event_base事件处理器对象的指针。
												s,	//表示要关联到bufferevent的已经打开的socket文件描述符。
												BEV_OPT_CLOSE_ON_FREE);	//表示缓冲事件的配置选项。

	//添加监控事件
	bufferevent_enable(bev,EV_READ|EV_WRITE);

	bufferevent_setwatermark(bev,EV_READ,
		5,	//低水位 0就是无限制 默认是0
		0	//高水位 0就是无限制 默认是0
		);

	bufferevent_setwatermark(bev,EV_WRITE,
		5,	//低水位 0就是无限制 默认是0 缓冲数据低于5 写入回调被调用
		0	//高水位无效
		);

	//超时时间的设置
	timeval t1 = {6,0};
	bufferevent_set_timeouts(bev,&t1,0);

	//设置回调函数
	bufferevent_setcb(bev,read_cb,write_cb,event_cb,base);
}

int main(int argc,char *argv[])
{
#ifdef _WIN32 
	//初始化socket库
	WSADATA wsa;
	WSAStartup(MAKEWORD(2,2),&wsa);
#else
	//忽略管道信号,发送数据给已关闭的socket
	if (signal(SIGPIPE, SIG_IGN) == SIG_ERR)
		return 1;
#endif

	event_base *base = event_base_new();
	//创建网络服务器

	//设定监听的端口和地址
	sockaddr_in sin;
	memset(&sin,0,sizeof(sin));
	sin.sin_family = AF_INET;
	sin.sin_port = htons(5001);

	evconnlistener *ev = evconnlistener_new_bind(base,	//事件处理器所使用的 event_base 对象
		listen_cb,		//新连接到来时调用的回调函数;
		base,			//回调函数的参数arg
		LEV_OPT_REUSEABLE | LEV_OPT_CLOSE_ON_FREE, //传递给回调函数的参数指针;
		10,				//侦听套接字的挂起连接队列的最大长度
		(sockaddr*)&sin,	//结构体指针,表示要绑定的地址和端口号;
		sizeof(sin)	//地址结构体的长度。
	);

	//进入事件主循环
	event_base_dispatch(base);
	evconnlistener_free(ev);
	event_base_free(base);

    return 0;
}

请添加图片描述

5 bufferevent客户端接例子

#include <iostream>
#include <event2/event.h>
#include <event2/listener.h>
#include <event2/bufferevent.h>
#include <string.h>
#ifndef _WIN32
#include <signal.h>
#else
#endif
#include <string>
using namespace std;

static int sendCount = 0;

//错误,超时 (连接断开会进入)
void client_event_cb(bufferevent *be,short events,void *arg)
{
	cout<<"[client_E]"<<flush;	
	//读取超时时间发生后,数据读取停止
	if(events & BEV_EVENT_TIMEOUT && events & BEV_EVENT_READING)
	{
		cout<<"BEV_EVENT_READING BEV_EVENT_TIMEOUT"<<endl;
		//bufferevent_enable(be,EV_READ);
		bufferevent_free(be);
		return;
	}
	else if(events & BEV_EVENT_ERROR)
	{
		bufferevent_free(be);
		return;
	}
	//服务端的关闭事件
	if(events & BEV_EVENT_EOF)
	{
		cout<<"BEV_EVENT_EOF"<<endl;
		bufferevent_free(be);
	}
	if(events & BEV_EVENT_CONNECTED )
	{
		cout<<"BEV_EVENT_CONNECTED"<<endl;
		//触发write
		bufferevent_trigger(be,EV_WRITE,0);
	}
	
}
void client_write_cb(bufferevent *be,void *arg)
{
	cout<<"[client_W]"<<flush;
	FILE *fp = (FILE *)arg;
	char data[4096] = {0};
	int len = fread(data,1,sizeof(data)-1,fp);
	if(len<=0)
	{
		//读到结尾或者文件出错
		fclose(fp);
		//立刻清理,可能会造成缓冲数据没有发送结束
		//bufferevent_free(be);
		bufferevent_disable(be,EV_WRITE);
		return;
	}
	sendCount += len;
	//写入buffer
	bufferevent_write(be,data,len);

}

void client_read_cb(bufferevent *be,void *arg)
{
	cout<<"[client_R]"<<flush;
}
int main(int argc,char *argv[])
{
    
#ifdef _WIN32 
	//初始化socket库
	WSADATA wsa;
	WSAStartup(MAKEWORD(2,2),&wsa);
#else
	//忽略管道信号,发送数据给已关闭的socket
	if (signal(SIGPIPE, SIG_IGN) == SIG_ERR)
		return 1;
#endif
	event_base *base = event_base_new();
	//创建网络服务器

{
	//调用客户端代码
	//-1内部创建socket 
	bufferevent *bev = bufferevent_socket_new(base,-1,BEV_OPT_CLOSE_ON_FREE);
	sockaddr_in sin;
	memset(&sin,0,sizeof(sin));
	sin.sin_family = AF_INET;
	sin.sin_port = htons(5001);
	evutil_inet_pton(AF_INET,"127.0.0.1",&sin.sin_addr.s_addr);

	FILE *fp = fopen("test_buffer_client.cpp","rb");
	//设置回调函数
	bufferevent_setcb(bev,client_read_cb,client_write_cb,client_event_cb,fp);
	bufferevent_enable(bev,EV_READ|EV_WRITE);
	int re = bufferevent_socket_connect(bev,(sockaddr*)&sin,sizeof(sin));
	if(re == 0)
	{
		cout<<"connected"<<endl;
	}
}


	//进入事件主循环
	event_base_dispatch(base);
	event_base_free(base);
	
	return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/524961.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

全面解析Linux指令和权限管理

目录 一.指令再讲解1.时间相关的指令2.find等搜索指令与grep指令3.打包和压缩相关的指令4.一些其他指令与热键二.Linux权限1.Linux的权限管理2.文件类型与权限设置3.目录的权限与粘滞位 一.指令再讲解 1.时间相关的指令 date指令: date 用法&#xff1a;date [OPTION]… [FOR…

缓冲区的flip

流和缓冲区都是用来描述数据的。计算机中&#xff0c;数据往往会被抽象成流&#xff0c;然后传输。比如读取一个文件&#xff0c;数据会被抽象成文件流&#xff1b;播放一个视频&#xff0c;视频被抽象成视频流。处理节点为了防止过载&#xff0c;又会使用缓冲区削峰&#xff0…

巴西大神开发的 ARPL 黑群晖DSM系统引导在线编译工具

ARPL 是一款黑群晖系统引导在线编译工具&#xff0c;目前支持最新群晖系统DSM 7.1.1&#xff0c;今天为了折腾升级这个群晖系统DSM 7.1.1浪费了一天的时间&#xff0c;ARPL是巴西人一位大神开发的黑群晖系统引导在线编译工具&#xff0c;使用下来非常的不错&#xff0c;可惜没有…

3网络互联-3.4【实验】【计算机网络】

3网络互联-3.4【实验】【计算机网络】 前言推荐3网络互联3.4 IP分组转发与静态路由实验目的实验内容及实验环境实验原理1.路由器2.路由(Routing)3.IP分组的转发4.路由的构建5.静态路由设计原则 实验过程1&#xff0e;搭建一个仅包含直连路由的网络拓扑&#xff0c;观察路由器的…

时间序列分析

一、移动平均法 1.一次移动平均法 公式&#xff1a; 预测标准误差: 本质&#xff1a;用前N次数据预测t1期的数据 规律&#xff1a;如果实际数据波动较大&#xff0c;N值越大&#xff0c;预测到的数据波动越小 注意&#xff1a;一般不适用于波动较大的数据。用一次移动平均法…

Kafka原理之消费者

一、消费模式 1、pull(拉)模式(kafka采用这种方式) consumer采用从broker中主动拉取数据。 存在问题&#xff1a;如果kafka中没有数据&#xff0c;消费者可能会陷入循环中&#xff0c;一直返回空数据 2、push(推)模式 由broker决定消息发送频率&#xff0c;很难适应所有消费者…

【MySQL】 InnoDB

学习笔记&#xff0c;来源黑马程序员MySQL教程 文章目录 逻辑存储结构架构内存架构磁盘结构后台线程 事务原理概述redo logundo log MVCC基本概念实现原理1、隐藏字段2、undo log3、readview 总结 逻辑存储结构 一个表空间对应一张表一 页 对应B树上一个 节点Trx id&#xff1a…

Git cat命令的用法

cat (全称 concatenate) 命令是 Linux/类 Unix 操作系统中最常用的命令之一。cat 命令允许我们创建单个或多个文件、查看文件内容、连接文件和重定向终端或文件中的输出。 语法&#xff1a; cat [OPTION] [FILE]...1.终端查看一个文件内容 cat file01.txt2.终端查看多个文件…

熵、信息量、条件熵、联合熵、互信息简单介绍

熵、信息量、条件熵、联合熵、互信息简单介绍 近期在看对比学习论文&#xff0c;发现有不少方法使用了互信息这种方式进行约束&#xff0c;故在此整理一下网上查阅到的关于互信息的相关内容。 一、熵、信息量 关于熵的讨论&#xff0c;这个知乎专栏写的挺不错的。 熵在信息论…

【更新日志】填鸭表单TduckPro v5.1 更新

hi&#xff0c;各位Tducker小伙伴。 填鸭表单pro迎来了v5.1版本&#xff1b;本次我们进行了许多的功能新增和优化&#xff0c;能够让我们在日常使用中获得更好的体验。 让我们一起来康康新功能吧。 01 新增Pro功能 新增登录后才能填写表单。 新增表单卡片一键发布。 新增矩…

【C++学习】CC++内存管理

目录 一、C&C内存管理 二、C语言中动态内存管理方式&#xff1a;malloc/calloc/realloc/free 三、C内存管理方式 3.1 new/delete操作内置类型 3.2 new和delete操作符自定义类型 四、operator new与operator delete函数 4.1 operator new与operator delete函数&#x…

【云原生】使用外网Rancher2.5.12在阿里云自建内网K8s 1.20集群

目录 一、目标二、解决方案三、草图四、版本信息五、资源规划六、必要条件七、开始部署1、安装Docker2、安装Rancher3、解析Rancher Server URL域名4、创建K8s集群5、注册K8s集群节点 八、验证 一、目标 在云平台搭建一套高可用的K8s集群 二、解决方案 第一种&#xff1a;使…

横向移动-利用IPC$

环境主机 本次都是在内网自己搭的靶机实验 上线主机&#xff1a;windows2008R2 - 192.168.31.46 需要移动到的主机&#xff1a;windows2012 - 192.168.31.45 实验演示 1.确定域控 通过命令net time /domain&#xff0c;发现存在域 这里我们通过ping来发现域控的ip&#xff0c;…

UGUI Scroll Rect滚动矩形组件

1、概述 当需要在小区域显示占用大量空间的内容时&#xff0c;可以使用Scroll Rect。滚动矩形提供了滚动浏览此内容的功能。 通常&#xff0c;将Scroll Rect与Mask结合在一起以创建滚动视图&#xff0c;在该视图中&#xff0c;只有Scroll Rect内部的可滚动内容可见。它也可以…

类和对象【1】

全文目录 引言&#xff08;初识面向对象&#xff09;类和对象定义类访问限定及封装类定义的两种方式 类实例化与类对象大小this指针 总结 引言&#xff08;初识面向对象&#xff09; C语言是面向过程的&#xff0c;关注的是过程&#xff0c;分析出求解问题的步骤&#xff0c;通…

NSSCTF之Misc篇刷题记录⑩

NSSCTF之Misc篇刷题记录⑩ [CISCN 2022 初赛]ez_usb[SWPUCTF 2021 新生赛]你喜欢osu吗&#xff1f;[SWPUCTF 2021 新生赛]Bill[SWPUCTF 2021 新生赛]二维码不止有二维码[HGAME 2022 week1]好康的流量[红明谷CTF 2022]MissingFile[广东省大学生攻防大赛 2021]这是道签到题[羊城杯…

TOGAF架构开发方法—阶段 F:迁移规划

本章介绍迁移规划;也就是说&#xff0c;如何通过最终确定一个 详细的实施和迁移计划。 一、目标 F阶段的目标是&#xff1a; 最终确定架构路线图以及支持实施和迁移计划确保实施和迁移计划与企业的管理和实施方法相协调 企业整体变更组合的变化确保关键利益相关者了解工作包和…

【什么是蜂窝移动网络】

从 DataReportal 2021 年 1 月的统计数据来看&#xff0c;全球 78 亿人口中&#xff0c;有 52 亿手机用户&#xff0c;46 亿互联网用户。能够接入网络的设备越来越多&#xff0c;体量越来越大&#xff0c;不知道你有没有好奇过&#xff0c;这样一个庞大的世界是如何被构造出来的…

【Linux】指令(下)

⭐博客主页&#xff1a;️CS semi主页 ⭐欢迎关注&#xff1a;点赞收藏留言 ⭐系列专栏&#xff1a;Linux ⭐代码仓库&#xff1a;Linux 家人们更新不易&#xff0c;你们的点赞和关注对我而言十分重要&#xff0c;友友们麻烦多多点赞&#xff0b;关注&#xff0c;你们的支持是我…

论文阅读:Multimodal Graph Transformer for Multimodal Question Answering

文章目录 论文链接摘要1 contribution3 Multimodal Graph Transformer3.1 Background on Transformers3.2 Framework overview 框架概述3.3 Multimodal graph construction多模态图的构建Text graphSemantic graphDense region graph Graph-involved quasi-attention 总结 论文…