libevent高并发网络编程 - 02_libevent缓冲IO之bufferevent
文章目录
- 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 的事件处理器中,等待客户端连接。
具体来说,它可以用于实现以下功能:
- TCP 服务器:通过调用
evconnlistener_new_bind()函数创建一个监听套接字,然后使用listen()函数开始监听客户端连接请求。当客户端连接到达时,libevent将会自动触发相应的事件回调函数,可以在回调函数中进行新连接的处理和数据读写等操作。 - UDP 服务器:虽然 UDP 是无连接的协议,但同样可以使用
evconnlistener_new_bind()函数创建一个本地套接字并绑定到指定地址和端口上,以接收远程主机发送的 UDP 数据包。当有数据包到达时,libevent将会触发相应的事件回调函数,开发者可以在回调函数中进行数据包的处理和响应等操作。 - 其他网络应用:除了常见的 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_TIMEOUTBEV_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:表示回调函数的上下文参数,在调用回调函数时会传递给它。
readcb、writecb和eventcb这三个回调函数形式,如下所示:
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:表示高水位线的大小,当缓冲区中数据量超过该值时会暂停读取事件或可写事件。
如果不想使用水位线机制,可以将lowmark和highmark参数都设为0。另外,bufferevent_setwatermark()函数只对BEV_READ和BEV_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()函数的作用是将一个IPv4或IPv6地址字符串转换为网络字节序格式的二进制地址。该函数们在网络编程中,方便地进行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,并置errno为EINVAL等错误码。
例如,以下代码可以将一个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_FREEbufferevent_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时关闭socketbufferevent *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就是无限制 默认是00 //高水位 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//忽略管道信号,发送数据给已关闭的socketif (signal(SIGPIPE, SIG_IGN) == SIG_ERR)return 1;
#endifevent_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, //回调函数的参数argLEV_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;//触发writebufferevent_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;//写入bufferbufferevent_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//忽略管道信号,发送数据给已关闭的socketif (signal(SIGPIPE, SIG_IGN) == SIG_ERR)return 1;
#endifevent_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;
}
相关文章:
libevent高并发网络编程 - 02_libevent缓冲IO之bufferevent
文章目录 1. 为什么需要缓冲区?2. 水位3. bufferevent常用API3.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(…...
院内导航移动导诊服务体系,院内导航怎么实现?
院内导航怎么实现?经过多年发展,医院规模愈加庞大,尤其是综合性医院,院区面积较大,门诊、医技、住院等大楼及楼区内部设计复杂,科室、诊室数量众多,对于新患者犹如进入了迷宫,客观环…...
MCTP协议和NCSI
MCTP(Management Component Transport Protocol)是一种管理组件传输协议,用于在计算机系统中管理各种组件,例如固件、BIOS、操作系统等。MCTP 协议定义了一种传输格式,以便在各种总线上进行通信,例如 PCIe、…...
Jmeter接口测试流程详解
1、jmeter简介 Jmeter是由Apache公司开发的java开源项目,所以想要使用它必须基于java环境才可以; Jmeter采用多线程,允许通过多个线程并发取样或通过独立的线程对不同的功能同时取样。 2、jmeter安装 首先需要安装jdk(最好是最…...
怎样使用Web自动化测试减少手动劳动?以百度网站为例
从入门到精通!企业级接口自动化测试实战,详细教学!(自学必备视频) 目录 摘要 步骤1:安装和配置Selenium 步骤2:启动浏览器并访问百度网站 步骤3:关闭浏览器 总结 摘要 本指南将…...
union和位域的混合使用
1、union(共用体) 1.1、概述 C 语言中,union是一种数据类型,对比于结构体,结构体中的每个成员都占用独立的内存空间,而联合中所有的成员都共享同一个内存空间。 也就是说,union中的不同成员要…...
PMP 高项 07-项目质量管理
项目质量管理 概念 质量的基本概念 克劳斯比:符合要求 戴明:低成本条件下可预测的一致性和可靠度,适应市场需要 朱兰:适用性,满足客户需要 国际标准化组织:质量是反映实体(产品、过程或活动等…...
鸿蒙Hi3861学习十一-Huawei LiteOS-M(内存池)
一、简介 LiteOS将内核与内存管理分开实现,操作系统内核仅规定了必要的内存管理函数原型,而不关心这些内存管理函数是如何实现的。 LiteOS内存管理模块管理系统的内存资源,包括:初始化、分配、释放。 不采用C标准库中的内存管理函…...
MySQL原理(七):内存管理和磁盘管理
前言 上一篇介绍了 MySQL 的日志,这一篇将介绍内存管理和磁盘管理相关的内容。 内存管理 MySQL 的数据都是存在磁盘中的,我们要更新一条记录的时候,得先要从磁盘读取该记录,然后在内存中修改这条记录。修改完这条记录后会缓存起…...
【Shell脚本】Linux安装Nginx以及开机自启
目录 一、Linux安装Nginx脚本1、把编写好的安装Nginx脚本放置到nginx.sh文件中2、在检查网络的时候,这里的IP地址,填写的需要安装Nginx服务器的IP地址3、这里的端口号可按照自己的需要进行修改4、安装Nginx脚本 二、Nginx开机自启 一、Linux安装Nginx脚本…...
solidworks三维建模竞赛练习题
solidworks三维建模竞赛练习题:3D01‐ 01 solidworks三维建模竞赛练习题:3D01‐ 02 solidworks三维建模竞赛练习题:3D01‐ 03 solidworks三维建模竞赛练习题:3D01‐ 04 solidworks三维建模竞赛练习题:3D01‐ 05 solidw…...
Redis---订阅和发布
目录 消息系统命令 消息系统 发布/订阅,即 pub/sub,是一种消息通信模式:发布者也称为消息生产者,生产和发送消息到存储系统;订阅者也称为消息消费者,从存储系统接收和消费消息。这个存储系统可以是文件系…...
使用Statsmodel进行假设检验和线性回归
如果你使用 Python 处理数据,你可能听说过 statsmodel 库。Statsmodels 是一个 Python 模块,它提供各种统计模型和函数来探索、分析和可视化数据。该库广泛用于学术研究、金融和数据科学。在本文中,我们将介绍 statsmodel 库的基础知识、如何…...
mac电脑 安装homebrew、nvm、node、nrm
安装homebrew /bin/zsh -c "$(curl -fsSL https://gitee.com/cunkai/HomebrewCN/raw/master/Homebrew.sh)brew -v 查看版本号 根据提示。继续执行 不然之后安装nvm报错(Error: Command failed with exit 128: git);例子: g…...
chrome 113 因为策略原因 cookie显示非常隐蔽
难受的要死,找不到cookie,安装老版本还疯狂偷跑我代理的流量更新!!!!!!! 解决方案: 回退112 !!!!࿰…...
Python3 operator 模块
Python3 中的 operator 模块提供了一系列用于操作 Python 内置类型的函数,这些函数可以替代常规的操作符,同时也提供了一些额外的功能。下面是 operator 模块中的一些常用函数及其作用: 1. operator.add(a, b):返回 a 和 b 的和&…...
106.(cesium篇)cesium椎体旋转
听老人家说:多看美女会长寿 地图之家总目录(订阅之前建议先查看该博客) 文章末尾处提供保证可运行完整代码包,运行如有问题,可“私信”博主。 效果如下所示: 下面献上完整代码,代码重要位置会做相应解释 <html lang="en"> <...
springboot+vue漫画之家系统(源码+文档)
风定落花生,歌声逐流水,大家好我是风歌,混迹在java圈的辛苦码农。今天要和大家聊的是一款基于springboot的漫画之家系统。项目源码以及部署相关请联系风歌,文末附上联系信息 。 💕💕作者:风歌&a…...
什么是中国版软件能力成熟度之CSMM
当前,中国软件产业蓬勃发展,产业增速迅猛,发展韧性和潜力进一步凸显。由于我国长期缺乏相关的自主标准,20多年来国外标准在我国软件评估领域占据主导地位,不仅阻碍了我国自主产业发展,甚至通过评估等手段可…...
Jupyter Notebook环境如何搭建以及应用呢?
Jupyter Notebook 是一个开源的 Web 应用程序,可以创建和共享文学化程序文档,支持多种编程语言,如 Python、R、Julia等。下面是搭建 Jupyter Notebook 环境及应用的步骤: 安装 Python 首先需要在本地安装 Python。可以从 Python…...
KubeSphere 容器平台高可用:环境搭建与可视化操作指南
Linux_k8s篇 欢迎来到Linux的世界,看笔记好好学多敲多打,每个人都是大神! 题目:KubeSphere 容器平台高可用:环境搭建与可视化操作指南 版本号: 1.0,0 作者: 老王要学习 日期: 2025.06.05 适用环境: Ubuntu22 文档说…...
Spring Boot 实现流式响应(兼容 2.7.x)
在实际开发中,我们可能会遇到一些流式数据处理的场景,比如接收来自上游接口的 Server-Sent Events(SSE) 或 流式 JSON 内容,并将其原样中转给前端页面或客户端。这种情况下,传统的 RestTemplate 缓存机制会…...
练习(含atoi的模拟实现,自定义类型等练习)
一、结构体大小的计算及位段 (结构体大小计算及位段 详解请看:自定义类型:结构体进阶-CSDN博客) 1.在32位系统环境,编译选项为4字节对齐,那么sizeof(A)和sizeof(B)是多少? #pragma pack(4)st…...
CMake基础:构建流程详解
目录 1.CMake构建过程的基本流程 2.CMake构建的具体步骤 2.1.创建构建目录 2.2.使用 CMake 生成构建文件 2.3.编译和构建 2.4.清理构建文件 2.5.重新配置和构建 3.跨平台构建示例 4.工具链与交叉编译 5.CMake构建后的项目结构解析 5.1.CMake构建后的目录结构 5.2.构…...
C++ 基础特性深度解析
目录 引言 一、命名空间(namespace) C 中的命名空间 与 C 语言的对比 二、缺省参数 C 中的缺省参数 与 C 语言的对比 三、引用(reference) C 中的引用 与 C 语言的对比 四、inline(内联函数…...
【git】把本地更改提交远程新分支feature_g
创建并切换新分支 git checkout -b feature_g 添加并提交更改 git add . git commit -m “实现图片上传功能” 推送到远程 git push -u origin feature_g...
OpenPrompt 和直接对提示词的嵌入向量进行训练有什么区别
OpenPrompt 和直接对提示词的嵌入向量进行训练有什么区别 直接训练提示词嵌入向量的核心区别 您提到的代码: prompt_embedding = initial_embedding.clone().requires_grad_(True) optimizer = torch.optim.Adam([prompt_embedding...
以光量子为例,详解量子获取方式
光量子技术获取量子比特可在室温下进行。该方式有望通过与名为硅光子学(silicon photonics)的光波导(optical waveguide)芯片制造技术和光纤等光通信技术相结合来实现量子计算机。量子力学中,光既是波又是粒子。光子本…...
【笔记】WSL 中 Rust 安装与测试完整记录
#工作记录 WSL 中 Rust 安装与测试完整记录 1. 运行环境 系统:Ubuntu 24.04 LTS (WSL2)架构:x86_64 (GNU/Linux)Rust 版本:rustc 1.87.0 (2025-05-09)Cargo 版本:cargo 1.87.0 (2025-05-06) 2. 安装 Rust 2.1 使用 Rust 官方安…...
打手机检测算法AI智能分析网关V4守护公共/工业/医疗等多场景安全应用
一、方案背景 在现代生产与生活场景中,如工厂高危作业区、医院手术室、公共场景等,人员违规打手机的行为潜藏着巨大风险。传统依靠人工巡查的监管方式,存在效率低、覆盖面不足、判断主观性强等问题,难以满足对人员打手机行为精…...
