libevent高并发网络编程 - 01_libevent事件Event处理

news2025/1/10 20:45:19

文章目录

    • 1. libevent事件驱动和事件处理简介
    • 2. 事件状态分析
    • 3. 事件Event常用API
      • 3.1 event_base_new()
      • 3.2 event_base_free()
      • 3.3 event_new()
      • 3.4 event_add()
      • 3.5 event_del()
      • 3.6 event_free()
      • 3.7 event_base_dispatch()
      • 3.8 event_base_loopbreak()
      • 3.9 evsignal_new()
      • 3.10 event_self_cbarg()
      • 3.11 evsignal_pending()
      • 3.12 evutil_make_socket_nonblocking()
      • 3.13 evutil_make_listen_socket_reuseable()
    • 4. linux信号事件示例
    • 5. 定时器事件示例
    • 6. 网络服务器事件示例

1. libevent事件驱动和事件处理简介

libevent是一个事件驱动框架,可以用于处理网络通信等I/O操作其中,事件处理(Event)和缓冲区( bufferevent)处理是两个不同的组件。

  • 事件处理(event)主要负责监听文件描述符上的事件,并在事件发生时调用相应的回调函数进行处理例如,一个TCP连接建立成功后,事件处理器可以监听该连接上是否有可读、可写或异常事件发生,并在事件发生时触发相应的回调函数进行数据处理。

  • 缓冲区处理(bufferevent)则是对事件处理的补充,它提供了对I/O数据的缓冲和处理功能当事件处理器检测到某个文件描述符可读时,缓冲区处理器会从该文件描述符读取一定量的数据并将其缓存在内存中,然后通过回调函数对缓存数据进行处理。类似地,当事件处理器检测到某个文件描述符可写时,缓冲区处理器会将已缓存的数据写入文件描述符中。

因此,事件处理和缓冲区处理是两个不同的组件,但常常会结合使用,以实现高效的网络通信。

下面以libevent事件处理(event)学习,相关的API接口,编写linux信号事件示例 、定时器事件示例、网络服务器事件示例。

2. 事件状态分析

在这里插入图片描述

在libevent中,状态的切换是由不同的API调用触发的:

  1. initialized 已初始化:事件初始化通常使用event_new()evtimer_new()等函数创建一个新的事件对象,并设置其回调函数和关注的事件类型。
  2. pending 待决:将事件添加到事件处理器中,可以使用event_add()event_assign()函数,将事件添加到指定的事件处理器中等待处理。
  3. active 激活当事件处理器检测到相应的I/O事件时,会激活相关的事件并执行其回调函数事件处理器可使用event_base_loop()event_base_dispatch()等函数循环等待I/O事件的发生。
  4. persistent 持久的:如果需要将事件设置为永久性事件,可以在事件处理器中将其标记为永久性例如,在调用event_add()函数之前,可以使用event_set()函数设置EV_PERSIST标志来使事件变为永久性事件。

3. 事件Event常用API

3.1 event_base_new()

event_base_new()**用于创建一个新的事件处理器对象(event_base)**它没有任何参数,返回一个指向新创建的event_base对象的指针。

在使用libevent时,需要先创建一个事件处理器对象,然后将需要处理的事件添加到该事件处理器中。event_base_new()函数就是用来创建这个事件处理器对象的。例如:

struct event_base *base = event_base_new();

3.2 event_base_free()

event_base_free()用于释放事件处理器对象(event_base)所占用的资源它有一个参数,即需要释放的event_base对象的指针。

在使用libevent时,当不再需要某个事件处理器对象时,需要使用event_base_free()函数释放相应的资源。例如:

struct event_base *base = event_base_new();
// 使用base处理事件...
event_base_free(base);

3.3 event_new()

event_new()用于创建一个新的事件对象(struct event):

struct event * 
event_new(struct event_base *base, evutil_socket_t fd, short events, void (*cb)(evutil_socket_t, short, void *), void *arg)
    
参数
	base:指向事件处理器对象(event_base)的指针,表示新创建的事件对象将与该事件处理器相关联。
	fd:表示该事件对象关注的文件描述符。
	events:表示该事件对象关注的事件类型,可选值包括EV_READ、EV_WRITE、EV_SIGNAL等。
	callback:表示该事件对象在触发时需要执行的回调函数。
    arg:回调函数的传入参数。
    
返回一个指向新创建的struct event对象的指针

events参数可以用来指定一个事件(struct event)关注的事件类型。可选的事件类型包括:

  • EV_TIMEOUT:超时事件,当计时器到达指定时间时触发。
  • EV_READ:读事件,当文件描述符上有可读数据时触发。
  • EV_WRITE:写事件,当文件描述符可写入数据时触发。
  • EV_SIGNAL:信号事件,当指定的信号被触发时触发。
  • EV_PERSIST:持久事件,使得事件对象在处理完毕后不会自动删除,可以用于周期性地执行某个任务。

这些事件类型可以通过按位或运算组合使用,表示事件对象同时关注多个事件类型。例如,将一个事件对象设置为关注读、写和超时事件,可以使用以下代码:

short events = EV_READ | EV_WRITE | EV_TIMEOUT;

关注不同的事件类型可以实现各种不同的功能,例如:

  • 读事件:可以用来处理网络通信中的数据接收。
  • 写事件:可以用来处理网络通信中的数据发送。
  • 超时事件:可以用来实现定时任务。
  • 信号事件:可以用来捕获操作系统发送的信号并进行处理。
  • 永久事件:可以用来周期性地执行某些任务,例如定时检查某个状态或发送心跳包。

事件对象的回调函数(callback

void (*cb)(evutil_socket_t, short, void *)

参数:
	fd:触发事件的文件描述符。
	events:触发的事件类型,可以是EV_READ、EV_WRITE、EV_TIMEOUT等。
	arg:回调函数所关联的参数,在创建事件对象时通过arg参数传递给事件对象。

例如,在以下代码中,my_callback()函数是一个事件对象的回调函数,其参数为fdeventsarg

void my_callback(evutil_socket_t fd, short events, void *arg) {
    // 在此处编写事件回调函数的代码...
}

struct event_base *base = event_base_new();
int fd = ... ; // 一个文件描述符
short events = EV_READ | EV_PERSIST;
struct event *ev = event_new(base, fd, events, my_callback, NULL);

此代码会创建一个新的事件对象,并将其与上面创建的事件处理器对象相关联。事件对象关注的文件描述符为fd关注的事件类型为EV_READ并且设置为永久事件(即EV_PERSIST标志),当该文件描述符上有读事件发生时,就会调用my_callback()函数进行处理。

3.4 event_add()

event_add()用于将事件对象(struct event)添加到指定的事件处理器(event_base)中等待触发。

int event_add(struct event *ev, const struct timeval *tv)
    
	ev:需要添加到事件处理器中的事件对象的指针。
	timeout:表示等待事件触发的超时时间,可以为NULL表示不进行超时等待。当timeout参数为0时,表示事件处理器在等待事件触发时不会进行任何超时等待。程序会一直阻塞在event_base_dispatch()函数调用中,直到有事件触发或者调用了event_base_loopbreak()函数或event_base_loopexit()函数来终止事件循环。

例如,以下代码会将一个事件对象添加到指定的事件处理器中等待触发:

struct event_base *base = event_base_new();
int fd = ... ; // 一个文件描述符
short events = EV_READ | EV_PERSIST;
struct event *ev = event_new(base, fd, events, my_callback, NULL);

// 将事件对象添加到事件处理器中等待触发
struct timeval timeout = {5, 0}; // 设定5秒超时等待
event_add(ev, &timeout);

// 使用事件处理器等待事件触发
event_base_dispatch(base);

先创建了一个新的事件对象ev,并将其添加到指定的事件处理器base中,并指定了5秒的超时等待时间。在调用event_base_dispatch()函数时,程序将进入阻塞状态,并等待事件处理器中的事件触发。如果在等待期间没有任何事件触发,程序将在超时后自动返回。

3.5 event_del()

event_del()用于将事件对象(struct event)从事件处理器(event_base)中删除,使其不再等待触发。

int event_del(struct event *ev)
    
ev:需要从事件处理器中删除的事件对象的指针。

3.6 event_free()

event_free()用于释放事件对象(struct event)所占用的资源。它有一个参数,即需要释放的struct event对象的指针。

在使用libevent时,当不再需要某个事件对象时,需要使用event_free()函数释放相应的资源。例如:

struct event_base *base = event_base_new();
int fd = ... ; // 一个文件描述符
short events = EV_READ | EV_PERSIST;
struct event *ev = event_new(base, fd, events, my_callback, NULL);

// 使用ev处理事件...
event_free(ev);

3.7 event_base_dispatch()

event_base_dispatch()是libevent库中的一个函数,用于启动事件处理器(event_base)并进入事件循环,等待事件对象触发。

int event_base_dispatch(struct event_base *event_base)
{
	return (event_base_loop(event_base, 0));
}

3.8 event_base_loopbreak()

event_base_loopbreak()用于终止事件处理器(event_base)的事件循环。

int event_base_loopbreak(struct event_base *event_base)
    
base:需要终止事件循环的事件处理器对象的指针。

3.9 evsignal_new()

evsignal_new()用于创建一个新的信号事件对象(struct event)。

#define evsignal_new(b, x, cb, arg)				\
	event_new((b), (x), EV_SIGNAL|EV_PERSIST, (cb), (arg))

隐藏的事件 EV_SIGNAL|EV_PERSIST(信号事件|持久事件)

参数
    b:指向事件处理器对象(event_base)的指针,表示新创建的事件对象将与该事件处理器相关联。
    x:表示需要关注的信号编号。
    cb:表示该事件对象在触发时需要执行的回调函数。
    arg:回调函数的传入参数。
    
返回一个指向新创建的`struct event`对象的指针。
    
每个信号都有一个唯一的编号(signal number),例如:
    SIGINT:中断信号,通常由CTRL-C键盘按键触发。
    SIGTERM:终止信号,通常用于请求终止进程。
    SIGKILL:强制终止信号,无法被阻塞、处理或忽略。
    SIGPIPE:管道破裂信号,通常由写入已关闭的管道或socket时触发。
    SIGALRM:定时器到期信号,通常用于实现计时器功能。
    SIGHUP:重启信号,当控制终端或SSH连接关闭时触发。
void my_signal_callback(evutil_socket_t fd, short events, void *arg) {
    // 在此处编写信号事件回调函数的代码...
}

struct event_base *base = event_base_new();
int signum = SIGINT;
struct event *ev = evsignal_new(base, signum, my_signal_callback, NULL);

此代码会创建一个新的信号事件对象,并将其与上面创建的事件处理器对象相关联。信号事件对象关注的信号编号为SIGINT,当该信号被触发时,就会调用my_signal_callback()函数进行处理。

3.10 event_self_cbarg()

event_self_cbarg(),**获取当前正在执行回调函数的事件对象。它返回一个void指针,指向当前事件对象的起始地址。**如果在非回调函数中调用此函数,则返回NULL。

void * event_self_cbarg(void)
{
	return &event_self_cbarg_ptr_;
}

3.11 evsignal_pending()

evsignal_pending(), 用于检测指定信号是否处于非待决状态(non-pending state)的函数。如果指定信号处于非待决状态,则返回1;否则返回0。

#define evsignal_pending(ev, tv)	event_pending((ev), EV_SIGNAL, (tv))

3.12 evutil_make_socket_nonblocking()

evutil_make_socket_nonblocking()用于将 socket 设置为非阻塞模式。该函数的声明如下:

int evutil_make_socket_nonblocking(evutil_socket_t fd)

该函数的实现原理是通过调用 fcntl() 或者 ioctlsocket() 来将 socket 设置为非阻塞模式。如果操作成功,则返回 0;否则返回一个负数错误码。

3.13 evutil_make_listen_socket_reuseable()

evutil_make_listen_socket_reuseable() 用于设置 socket 地址重用选项。该函数的声明如下:

int evutil_make_listen_socket_reuseable(evutil_socket_t sock)

在 TCP 协议中,默认情况下,在端口被占用后,如果再次绑定该端口会返回错误。设置地址重用选项可以允许多个进程或线程同时监听同一端口号。

该函数的实现原理是通过调用 setsockopt() 函数来设置 SO_REUSEADDR 选项。如果操作成功,则返回 0;否则返回一个负数错误码。

4. linux信号事件示例

#include <iostream>
#include <event2/event.h>
#include <signal.h>
using namespace std;

//sock 文件描述符,which 事件类型 arg传递的参数
static void Ctrl_C(evutil_socket_t sock, short which, void *arg)
{
    cout<<"Ctrl_C"<<endl;
}

static void Kill(evutil_socket_t sock, short which, void *arg)
{
    cout<<"Kill"<<endl;
    event *ev = (event*)arg;

    //如果处于非待决
    if(!evsignal_pending(ev, NULL))
    {
        event_del(ev);
		event_add(ev,NULL);
    }
}

int main(int argc,char *argv[])
{
    //创建一个新的事件处理器对象
    event_base *base = event_base_new();

    //添加ctrl +C 信号事件,处于no pending
    //evsignal_new 隐藏的事件 EV_SIGNAL|EV_PERSIST(信号事件|持久事件)
    event* csig = evsignal_new(base, SIGINT, Ctrl_C, base);
    if(!csig)
    {
        cerr<<"SIGINT evsignal_new failed!"<<endl;
        return -1;
    }
    //添加事件到pending
    if(event_add(csig,0) != 0)
    {
        cerr<<"SIGINT event_add failed!"<<endl;
        return -1;
    }


    //添加kill信号
    //非持久事件,只进入一次 event_self_cbarg()传递当前的event
    event *ksig = event_new(base, SIGTERM, EV_SIGNAL, Kill, event_self_cbarg());
    if(!ksig)
    {
        cerr<<"SIGTERM evsignal_new failed!"<<endl;
        return -1;
    }
    //添加事件到pending
    if(event_add(ksig,0) != 0)
    {
        cerr<<"SIGTERM event_add failed!"<<endl;
        return -1;
    }

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

    //释放事件对象
    event_free(csig);

    //释放事件处理器对象
    event_base_free(base);

    return 0;
}

请添加图片描述

5. 定时器事件示例

#include <iostream>
#include <event2/event.h>
#ifndef _WIN32
#include <signal.h>
#else
#endif

using namespace std;

static timeval t1 = {1,0};

void timer1(int sock,short which,void *arg)
{
	cout<<"[timer1]"<<flush;
	event *ev = (event *)arg;
	//no pending
	if(!evtimer_pending(ev,&t1))
	{
		evtimer_del(ev);
		evtimer_add(ev,&t1);
	}
}

void timer2(int sock,short which,void *arg)
{
	cout<<"[timer2]"<<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();
	
	//定时器
	cout<<"test timer"<<endl;
	
	//event_new
	/*
	#define evtimer_new(b, cb, arg)		event_new((b), -1, 0, (cb), (arg))
	#define evtimer_add(ev, tv)		event_add((ev), (tv))
	#define evtimer_del(ev)			event_del(ev)
	*/
	//定时器 非持久事件 
    event *ev1 = evtimer_new(base,timer1,event_self_cbarg());
	if(!ev1)
	{
		cout<<"evtimer_new timer1 failed!"<<endl;
		return -1;
	}
	evtimer_add(ev1,&t1); //插入性能 O(logn)

    static timeval t2;
	t2.tv_sec = 2;
	t2.tv_usec = 200000; //微秒
	event *ev2 = event_new(base,-1,EV_PERSIST,timer2,0);
	event_add(ev2,&t2);



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

请添加图片描述

6. 网络服务器事件示例

#include <iostream>
#include <event2/event.h>

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

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

using namespace std;
#define SPORT 5001 

//正常断开连接也会进入,超时会进入
void client_cb(evutil_socket_t s,short w, void *arg)
{
	//水平触发LT 只有有数据没有处理,会一直进入
	//边缘触发ET 有数据时只进入一次
	//cout<<"."<<flush;return;
	
	event *ev = (event *)arg;
	//判断超时
	if(w&EV_TIMEOUT)
	{
		cout<<"timeout"<<endl;
		event_free(ev);
		evutil_closesocket(s);
		return;
	}
	
	//char buf[1024] = {0};
	char buf[1024] = {0};
	int len = recv(s,buf,sizeof(buf)-1,0);
	if(len>0)
	{
		cout<<buf;
		send(s,"ok",2,0);
	}
	else
	{
		//需要清理event
		cout<<"event_free"<<flush;
		event_free(ev);
		evutil_closesocket(s);
		
	}
	
}

void listen_cb(evutil_socket_t s,short w, void *arg)
{
	cout<<"listen_cb"<<endl;
	sockaddr_in sin;
	socklen_t size = sizeof(sin);
	//读取连接信息
	evutil_socket_t client = accept(s,(sockaddr*)&sin,&size);
	char ip[16] = {0};
	evutil_inet_ntop(AF_INET,&sin.sin_addr,ip,sizeof(ip)-1);
	cout<<"client ip is "<<ip<<endl;
	
	//客户端数据读取事件
	event_base *base = (event_base *)arg;
	event *ev = event_new(base,client,EV_READ|EV_PERSIST,client_cb,event_self_cbarg());
	//边缘触发
	//event *ev = event_new(base,client,EV_READ|EV_PERSIST|EV_ET,client_cb,event_self_cbarg());
	timeval t = {10,0};
	event_add(ev,&t);
}

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();
	
	//event 服务器
	cout<<"test event server"<<endl;
	
	//创建socket
	evutil_socket_t sock = socket(AF_INET,SOCK_STREAM,0);
	if(sock<=0)
	{
		cerr<<"socket error:"<<strerror(errno)<<endl;
		return -1;
	}

	//设置地址复用和非阻塞
	evutil_make_socket_nonblocking(sock);
	evutil_make_listen_socket_reuseable(sock);

	//绑定端口和地址
	sockaddr_in sin;
	memset(&sin,0,sizeof(sin));
	sin.sin_family = AF_INET;
	sin.sin_port = htons(SPORT);
	int re = ::bind(sock,(sockaddr*)&sin,sizeof(sin));  //使用全局bind,不使用std的bind
	if(re != 0)
	{
		cerr<<"bind error:"<<strerror(errno)<<endl;
		return -1;
	}
	//开始监听
	listen(sock,10);

	//开始接受连接事件 默认水平触发
	event *ev = event_new(base,sock,EV_READ|EV_PERSIST,listen_cb,base);
	event_add(ev,0);
	
	//进入事件主循环
	event_base_dispatch(base);
	evutil_closesocket(sock);
	event_base_free(base);
		
    return 0;
}

请添加图片描述

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

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

相关文章

继承与虚函数练习

Tip1 基类私有成员变量在子类中都不能直接访问&#xff0c;不是因为没有被子类继承&#xff0c;而是权限问题 Tip2 满足多态的父子对象&#xff0c;父类对象和子类对象前4个字节都是虚表指针&#xff08;vs2019下&#xff09;&#xff0c;父类与子类指向的是各自的虚表。 Tip…

云计算中的容器技术及其实践案例

第一章&#xff1a;什么是容器技术 随着云计算和DevOps的普及&#xff0c;容器技术在IT行业中越来越受到关注。容器是一种轻量级、可移植、可扩展的应用程序封装技术&#xff0c;可以将应用程序及其所有依赖项打包到一个独立的可执行文件中。相对于虚拟机技术&#xff0c;容器技…

无界AI绘画基础教程,和Midjourney以及Stable Diffusion哪个更好用?

本教程收集于&#xff1a;AIGC从入门到精通教程汇总 简单的总结 Midjourney&#xff0c;Stable Diffusion&#xff0c;无界AI的区别&#xff1f; Midjourney&#xff0c;收费&#xff0c;上手容易&#xff0c;做出来高精度的图需要自己掌握好咒语。咒语写不好&#xff0c;像…

【LLM大模型】LLM模型指令微调(更新中)

note 文章目录 note零、AIGC生成式模型1. 核心要素2. LLM evolutionary tree 二、LLM大模型1. ChatGLM&#xff08;1&#xff09;GLM-130B&#xff08;2&#xff09;ChatGLM-6B 2. LLaMA3. Chinese-LLaMA-Alpace4. Bloom5. PaLM 三、模型指令微调Reference 零、AIGC生成式模型 …

算法记录 | Day53 动态规划

1143.最长公共子序列 思路&#xff1a; 本题和动态规划&#xff1a;718. 最长重复子数组 (opens new window)区别在于这里不要求是连续的了&#xff0c;但要有相对顺序&#xff0c;即&#xff1a;“ace” 是 “abcde” 的子序列&#xff0c;但 “aec” 不是 “abcde” 的子序…

搭建DHCP、PXE、DNS、HTTP以及NFS服务综合实验的超详细讲解

文章目录 1.实验要求2.实验步骤2.1 步骤解答问题&#xff08;1&#xff09;2.1 步骤解答问题&#xff08;2&#xff09;2.1 步骤解答问题&#xff08;3&#xff09;2.1 步骤解答问题&#xff08;4&#xff09;2.1 步骤解答问题&#xff08;5&#xff09; 1.实验要求 &#xff…

Hadoop入门篇01---基础概念和部署教程

Hadoop入门篇01---基础概念和部署教程 Hadoop是什么Hadoop发展史Hadoop特点有哪些Hadoop版本Hadoop架构Hadoop 3.0新特性 Hadoop集群搭建集群简介集群部署方式standalone mode&#xff08;独立模式&#xff09;Pseudo-Distributed mode&#xff08;伪分布式模式&#xff09;Clu…

webpack的核心概念分别是什么,如何理解

这篇文章主要介绍了 title &#xff0c;小编觉得挺不错的&#xff0c;现在分享给大家&#xff0c;也给大家做个参考&#xff0c;希望大家通过这篇文章可以有所收获。 webpack 是一种前端资源构建工具&#xff0c;一个静态模块打包器&#xff08;module bundler&#xff09;&…

UE5.1.1C++从0开始(4.虚幻的接口以及交互功能)

这一个章节对于第一次接触虚幻的人来说可以说是最绕的一个点&#xff0c;因为老师突然给你塞了很多的概念&#xff0c;对于这一块的学习&#xff0c;我个人的推荐是&#xff1a;先把蓝图搞明白了&#xff0c;再对应到C的代码中&#xff0c;不然一定会被整的晕头转向。还是&…

安装ms sql server2000提示安装失败详见sqlstp.log日志

安装ms sql server2000提示安装失败详见sqlstp.log日志 目录 安装ms sql server2000提示安装失败详见sqlstp.log日志 一、可能的情况-其它位置不能有对它的引用 1.1、先安装了Delphi其options-环境变量-其中path中有sql&#xff0c;注册表将其清除 1. 2、注册表搜索-Micro…

【Unity3d】Unity3d在Android平台上输入框的实现源码分析

一、前言 Unity3d引擎中有很多与Android原生交互的功能&#xff0c;比如如何调用Android渲染、Unity输入框的实现、Unity权限的申请、Unity调用相机功能等等&#xff0c;其实这些就是调用Android的api实现的。所有Java层的实现代码都是在unity-classes.jar这个jar包中。这篇文…

chatgpt的150个指令大全

chatGPT输出结果的质量高低&#xff0c;和你使用什么样质量的输入内容有关。 今天&#xff0c;小编整理了一些常用的ChatGPT指令&#xff0c;可以通过这些指令让AI帮你整理资料、撰写报告、知识学习、准备面试等等&#xff0c;赶紧收藏起来吧&#xff01; 如果还没有找到工具…

两万字详解!Netty经典32连问

两万字详解&#xff01;Netty经典32连问&#xff01; 前言 Netty是什么&#xff0c;它的主要特点是什么&#xff1f; Netty 应用场景了解么&#xff1f; Netty 核心组件有哪些&#xff1f;分别有什么作用&#xff1f; Netty的线程模型是怎样的&#xff1f;如何优化性能&…

四、Spring Cloud Alibaba-Ribbon

一、什么是Ribbon 目前主流的负载方案分为以下两种: 集中式负载均衡&#xff0c;在消费者和服务提供方中间使用独立的代理方式进行负载&#xff0c;有硬件的(比如 F5)&#xff0c;也有软件的(比如 Nginx) 。客户端根据自己的请求情况做负载均衡&#xff0c;Ribbon 就属于客户…

2023年全国硕士研究生入学统一考试英语(一)试题

2023年全国硕士研究生入学统一考试英语&#xff08;一&#xff09;试题 Section I Use of Englis Directions: Read the following text. Choose the best word(s) for each numbered blank and mark A, B , C or D on the ANSWER SHEET.(10 points) Caravanserais were roads…

【P9】JMeter 用户定义的变量(User Defined Variables)

一、准备工作 慕慕生鲜&#xff1a; http://111.231.103.117/#/login 进入网页后&#xff0c;登录&#xff0c;页面提供了账户和密码 右键检查或按F12&#xff0c;打开调试工具&#xff0c;点击搜索 二、测试计划设计 &#xff08;1&#xff09;、Test Plan 右键 <<…

Linux cgroup

前言 Cgroup和namespace类似&#xff0c;也是将进程进程分组&#xff0c;但是目的与namespace不一样&#xff0c;namespace是为了隔离进程组之前的资源&#xff0c;而Cgroup是为了对一组进程进行统一的资源监控和限制。 Cgroup的组成 subsystem 一个subsystem就是一个内核模…

【HCIP】VLAN实验(Hybrid模式)

目录 需求&#xff1a; 一、设计 二、VLAN配置 三、交换机间实现trunk的功能 四、路由器配置 五、验证 需求&#xff1a; 1、PC1和PC3所在接口为access 2、PC2/4/5/6处于同一网段&#xff0c;其中PC2可以访问PC4/5/6&#xff1b;但PC4可以访问PC5&#xff0c;不能访问PC…

ARM嵌入式编译器-volatile关键字对编译器优化的影响

volatile限定符告知计算机&#xff0c;其他agent&#xff08;而不是变量所在的程序&#xff09;可以改变该变量的值。通常它被用于硬件地址以及在其他程序或同时运行的线程中共享数据。要求编译器不要对其描述的对象作优化处理&#xff0c;对它的读写都需要从内存中访问。 使用…

文献阅读:LLaMA: Open and Efficient Foundation Language Models

文献阅读&#xff1a;LLaMA: Open and Efficient Foundation Language Models 1. 文章简介2. 模型训练 1. 训练数据2. 模型结构3. 模型训练 1. Optimizer2. 效率优化 3. 效果评估 1. 经典任务下效果 1. Commen Sense Reasoning2. Closed-book Question Answering3. Reading Co…