当前位置: 首页 > news >正文

Linux网络编程(十五)——优于select的epoll

文章目录

15 优于select的epoll

15.1 epoll理解及应用

15.1.1 基于select的I/O复用技术速度慢的原因

15.1.2 select的优点

15.1.3 实现epoll时必要的函数和结构体

15.1.4 epoll_creat1

15.1.5 epoll_ctl

15.1.6 epoll_wait

15.1.7 基于epoll的回声服务器端

15.2 条件触发和边缘触发

15.2.1 条件触发和边缘触发的区别在于发生事件的时间点

15.2.2 掌握条件触发的事件特性

15.2.3 边缘触发的服务器端实现中必知的两点

15.2.4 实现边缘触发的回声服务器端

15.2.5 边缘触发和条件触发孰优孰劣


15 优于select的epoll

15.1 epoll理解及应用

select复用方法由来已久,由于无法同时接入上百个客户端,所以并不适合以Web服务端开发为主流的现代开发环境,所以需要linux平台下的epoll

15.1.1 基于select的I/O复用技术速度慢的原因

第10章曾实现过基于select的I/O复用服务器端,很容易从代码上分析出不合理的设计,主要以下两点:

  • 调用select函数后常见的针对所有文件描述符的循环语句
  • 每次调用select函数时都需要向该函数传递监视对象信息

调用 select 函数后,并不是把发生变化的文件描述符单独集中到一起,而是通过观察作为监视对象的 fd_set 变量的变化,找出发生变化的文件描述符,因此无法避免针对所有监视对象的循环语句。而且,作为监视对象的 fd_set 变量会发生变化,所以每次循环调用 select 时都要传进去新的 fd_set 变量。

后者导致的性能问题无法避免,因为每次调用 select 函数时要向操作系统传递监视对象信息,会经历用户空间到内核空间的复制(FD_ISSET由操作系统判断)循环遍历时,还要将整个变量从内核空间复制到用户空间。如果只复制一次给操作系统,由操作系统监视,然后通知发送变化的事项就好了,所以Linux有了epoll。

15.1.2 select的优点

大部分操作系统都支持select函数,只要满足或要求如下两个条件,即使在Linux平台也不应拘泥于epoll

  • 服务器端接入者少
  • 程序应具有兼容性

15.1.3 实现epoll时必要的函数和结构体

能够克服select函数缺点的epoll函数具有以下优点,恰好与之前的select函数缺点相反。

  • 无需编写以监视状态变化为目的的针对所有文件描述符的循环语句。
  • 调用对应于select函数的epoll_wait函数时无需每次传递监视对象信息。

下面是实现epoll服务端需要的3个函数,结合epoll的优点理解这些函数的功能。

  • epoll_create1/epoll_create:创建保存 epoll 文件描述符的空间(epoll例程)
  • epoll_ctl:向空间注册并注销文件描述符。
  • epoll_wait:与select函数类似,等待文件描述符发生变化。
特性/功能select方式epoll方式
保存监视对象文件描述符直接声明 fd_set 变量使用 epoll_creat1/epoll_create 请求操作系统创建空间
添加/删除监视对象FD_SET 和 FD_CLR使用 epoll_ctl 函数请求操作系统完成
等待文件描述符变化调用 select 函数调用 epoll_wait 函数
查看状态变化通过 fd_set 查看监视对象状态变化通过 epoll_event结构体将发生变化的文件描述符单独集中到一起
事件处理方式每次调用需遍历所有文件描述符仅返回发生事件的文件描述符
适用场景适合少量文件描述符的场景适合大量文件描述符的高并发场景
struct epoll_event {epoll_data_t data;    //哪个文件描述符发生了变化(监视哪个文件描述符)__uint32_t events;    //出现了什么变化(监听文件描述符的什么事件)
/*            (1)EPOLLIN:需要读取数据的情况(可读取的数据)
*             (2)EPOLLOUT:输出缓冲为空,可以立即发送数据的情况
*             (3)EPOLLPRI:收到OOB数据的情况
*             (4)EPOLLRDHUB:断开连接或者半关闭的情况,这在边缘触发方式下非常有用
*             (5)EPOLLERR:发生错误的情况
*             (6)EPOLLET:以边缘触发方式得到事件通知
*             (7)EPOLLONESHOT:发生一次事件后,相应文件描述符不再收到事件通知。
*                    因此需要向epoll_ctl函数的第二个参数传递EPOLL_CTL_MOD,再设置事件
*/                  
}
typedef union epoll_data
{void* ptr;int fd;        //监视的文件描述符(通常只关注该成员)__uint32_t u32;__uint64_t u64;
}epoll_data_t;

15.1.4 epoll_creat1

#include <sys/epoll.h>int epoll_create1(int flag);
/*
* flag:如果填0,直接创建epoll,也可以标记获得不同的行为
* return:成功时返回epoll文件描述符,失败时返回-1
*/

调用 epoll_create1 函数时创建的文件描述符保存空间称为 "epoll例程"。epoll_ create1 函数创建的资源与套接字相同,也由操作系统管理。 因此,该函数也会返回文件描述符。 也就是说,该函数返回的文件描述符主要用与于区分 epoll 例程。 需要终止时与其他文件描述符相同,也要调用close函数。

15.1.5 epoll_ctl

生成 epoll 的例程后,接下来就是在其内部注册监视对象文件描述符。

int epoll_ctl(int epfd, int op, int fd, struct epoll_event* event);
/*
* epfd:epoll实例的文件描述符(需要监听的epoll池)
* op:用于指定监视对象的添加、删除或更改等操作
*            (1)EPOLL_CTL_ADD:将文件描述符添加到epoll例程
*            (2)EPOLL_CTL_DEL:删除
*            (3)EPOLL_CTL_MOD:更改注册的文件描述符的关注事件发生情况。
* fd:需要注册的监视对象文件描述符(需要监听的文件描述符)
* event:保存发生事件的文件描述符集合的结构体地址
* return:成功返回0,失败返回-1
*/

例如 epoll_ctl(A,EPOLL_CTL_ADD,B,C); 意思是 “在例程A中注册文件描述符B,主要目的是监视参数C中的事件”

15.1.6 epoll_wait

声明足够大的 epoll_even 结构体数组后,传递给 epoll_wait 函数时,发生变化的文件描述符信息将被填入该数组。 因此,无需像 select 函数那样针对所有文件描述符进行循环。

int epoll_wait(int epfd, struct epoll_event* events, int maxevents, int timeout);
/*
* epfd:epoll实例的文件描述符(需要监听的epoll池)
* events:提供给内核,用于返回已就绪的文件描述符的信息,空间需要动态分配
* maxevents:可以返回最大文件描述符数量
* timeout:超时时间,指明epoll_wait()在事件触发前阻塞等待最大秒数
*            -1表示一直等待至有事件发生,0表示无论是否有事件发生立即返回
* return:成功时返回发生变化的文件描述符数量,超时仍没有就绪事件返回0,失败返回-1
*/

15.1.7 基于epoll的回声服务器端

基于第10章I/O复用服务器端实现了该服务器端实例,客户端代码与第三章回声客户端代码相同。

#define BUF_SIZE 1024
#define EPOLL_SIZE 50
void error_handling(char *message) {fputs(message, stderr);fputc('\n', stderr);exit(1);
}int main(int argc, char *argv[])
{int serv_sock,clnt_sock;struct sockaddr_in serv_addr,clnt_addr;//epoll_ctl的参数struct epoll_event event;//epoll_wait的参数,因为需要动态创建所以声明成指针形式struct epoll_event* ep_events;      char* buf = malloc(sizeof(char)*BUF_SIZE);memset(&serv_addr,0,sizeof(serv_addr));memset(&clnt_addr,0,sizeof(clnt_addr));if(argc != 2) {printf("Usage:%s <port>\n",argv[0]);exit(1);}serv_addr.sin_family = AF_INET;serv_addr.sin_port = htons(atoi(argv[1]));inet_pton(AF_INET,"0.0.0.0",&serv_addr.sin_addr);serv_sock = socket(PF_INET, SOCK_STREAM, 0);//设置地址再分配int option = 1;setsockopt(serv_sock,SOL_SOCKET,SO_REUSEADDR,&option,sizeof(option));printf("服务端的文件描述符是:%d\n",serv_sock);if(bind(serv_sock, (struct sockaddr*)&serv_addr, sizeof(serv_addr)) == -1)error_handling("bind() error");if(listen(serv_sock, 5) == -1)error_handling("listen() error");//创建例程int epfd = epoll_create1(0);//为ep_events动态创建大小ep_events = malloc(sizeof(struct epoll_event*)&EPOLL_SIZE);//注册服务端文件描述符event.events = EPOLLIN;event.data.fd = serv_sock;//为例程epfd注册文件描述符serv_sock,主要目的是监视serv_sock中的EPOLLIN事件epoll_ctl(epfd,EPOLL_CTL_ADD,serv_sock,&event);while(1) {//等待文件描述符发生变化,成功则返回发生事件的文件描述符数int event_cnt = epoll_wait(epfd,ep_events,EPOLL_SIZE,-1);if(event_cnt==-1) {puts("epoll_wait() error");break;}//遍历发生变化的文件描述符集合for(int i = 0;i < event_cnt;i++) {//如果发生变化的是服务端文件描述符,则接受连接if(ep_events[i].data.fd == serv_sock) {socklen_t adr_sz = sizeof(clnt_addr);int clnt_sock = accept(serv_sock,(struct sockaddr*)&clnt_addr,&adr_sz);//注册客户端文件描述符event.events = EPOLLIN;event.data.fd = clnt_sock;epoll_ctl(epfd,EPOLL_CTL_ADD,clnt_sock,&event);printf("connected client:%d\n",clnt_sock);}//如果发生变化的是客户端文件描述符,收发消息else {//有可能与多个客户端相连并且客户端同时发送消息,所以可能同时变换很多个文件描述符//i指的是当前发起请求的客户端int str_len = recv(ep_events[i].data.fd,buf,BUF_SIZE,0);//连接正常关闭if(str_len==0) {//将该文件描述符从例程中移除epoll_ctl(epfd,EPOLL_CTL_DEL,ep_events[i].data.fd,NULL);close(ep_events[i].data.fd);printf("closed client:%d\n",ep_events[i].data.fd);}else {send(ep_events[i].data.fd,buf,str_len,0);}}}}close(serv_sock);close(epfd);free(buf);return 0;
}

15.2 条件触发和边缘触发

15.2.1 条件触发和边缘触发的区别在于发生事件的时间点

条件触发中,只要输入缓冲中有数据就会一直通知该事件。例如,服务器端输入缓冲收到50字节数据时,服务器端操作系统将通知该事件(注册到发生变化的文件描述符)。但服务器端读取20字节后还剩30字节的情况下,仍会注册事件。也就是说,条件触发方式中,只要输入缓冲中还剩余数据,就将以事件方式再次注册。

边缘触发中输入缓冲收到数据时仅注册1次该事件。即使输入缓冲中还留有数据,也不会再进行注册。

15.2.2 掌握条件触发的事件特性

epoll 默认以条件触发方式工作,因此,通过修改15.1.7小节服务器端代码验证条件触发特性

  • 将调用recv函数时使用的缓冲大小 BUF_SIZE 缩减为512字节
  • 在 epoll_wait 函数下方插入验证该函数调用次数的语句

减少缓冲大小是为了阻止服务器端一次性读取接收的数据。换言之,调用 recv 函数后,输人缓冲中仍有数据需要读取,因此会注册新的事件并从 epoll_wait 函数返回。

对于采用条件触发工作模式的文件描述符,当 epoll_wait 检测到其上有事件发生并将此事件通知应用程序后,应用程序可以不立即处理该事件。当应用程序下一次调用 epoll_wait 时,epoll_wai还会再次向应用程序通告此事件,直到此事件被处理。(意思是:这次没读完,下次接着读)

从运行结果中看,每当收到客户端数据时,应用程序没有立即完成对事件的处理(没有一次读完数据),因此,会将未完成事件重新注册到 “例程” 中,epoll_wait 重新监测到该事件并通告该事件让应用程序继续处理。下面将上述示例改成边缘触发方式,只需要在条件触发代码基础上修改如下代码即可

//注册客户端文件描述符
event.events = EPOLLIN|EPOLLET;

更改代码后,从客户端接收数据,仅输出1次 "return epoll_wait" 字符串,这意味着仅注册1次事件。但是客户端运行时发生错误,下一小节对错误进行分析。

15.2.3 边缘触发的服务器端实现中必知的两点

如下两点是实现边缘触发的必知内容

  • 通过errno变量验证错误原因
  • 为了完成非阻塞I/O,更改套接字特性

套接字相关函数在失败时通常会返回 -1,但这并不能直接提供失败的具体原因。为了提供额外的错误信息,Linux 使用了全局变量 errno。为了访问该变量,需要包含头文件 <errno.h>每个函数在发生错误时,会将不同的错误代码保存到 errno 变量中,具体的错误代码含义可以通过查阅相关资料得知,通常无需记住所有可能的错误值。

Linux提供了更改或读取文件属性的如下方法

#include <fcntl.h>int fcntl(int filedes, int cmd, ...);
/*
* filedes:属性更改目标的文件描述符
* cmd:执行的操作,常用的有F_GETFL和F_SETFL
*        F_GETFL:返回文件描述符的权限模式和状态标记,不需要额外参数
*        F_SETFL:返回文件的状态标记设置为第三个参数指定的值
* ...(可变参数):cmd需要的参数,可以没有
* return:成功时返回cmd参数相关值。失败返回-1
*/

如希望将文件修改为非阻塞模式,需要如下2条语句

int flag = fcntl(fd, F_GETFL, 0);  //获取当前文件的状态标志
fcntl(fd, F_SETFL, flag | O_NONBLOCK);  //设置文件状态标志

通过第一条语句获取之前设置的属性信息,通过第二条语句在此基础上添加非阻塞
O_NONBLOCK标志。调用read&write函数时,无论是否存在数据,都会形成非阻塞文件(套接
字)。

15.2.4 实现边缘触发的回声服务器端

(1)为什么需要errno确认错误原因?

“边缘触发方式中,接收数据时仅注册1次该事件” 就因为这种特点,一旦发生输入相关事件,就应该读取输入缓冲中全部数据,因此需要验证输入缓冲是否为空。

(2)为社么需要将套接字变成非阻塞模式?

边缘触发方式下,以阻塞方式工作的 recv&send 函数有可能引起服务器端的长时间停顿。因此,边缘触发方式中一定要采用非阻塞 recv&send 函数。

(3)以边缘触发方式工作的回声服务器端示例

#define BUF_SIZE 4
#define EPOLL_SIZE 50   //例程大小
void error_handling(char *message) {fputs(message, stderr);fputc('\n', stderr);exit(1);
}void setnonblockingmode(int fd) {int flag = fcntl(fd, F_GETFL, 0);fcntl(fd, F_SETFL, flag | O_NONBLOCK);
}int main(int argc, char *argv[])
{int serv_sock,clnt_sock;struct sockaddr_in serv_addr,clnt_addr;//epoll_ctl的参数struct epoll_event event;//epoll_wait的参数,因为需要动态创建所以声明成指针形式struct epoll_event* ep_events;      char* buf = malloc(sizeof(char)*BUF_SIZE);memset(&serv_addr,0,sizeof(serv_addr));memset(&clnt_addr,0,sizeof(clnt_addr));if(argc != 2) {printf("Usage:%s <port>\n",argv[0]);exit(1);}serv_addr.sin_family = AF_INET;serv_addr.sin_port = htons(atoi(argv[1]));inet_pton(AF_INET,"0.0.0.0",&serv_addr.sin_addr);serv_sock = socket(PF_INET, SOCK_STREAM, 0);//设置地址再分配int option = 1;setsockopt(serv_sock,SOL_SOCKET,SO_REUSEADDR,(void*)&option,sizeof(option));printf("服务端的文件描述符是:%d\n",serv_sock);if(bind(serv_sock, (struct sockaddr*)&serv_addr, sizeof(serv_addr)) == -1)error_handling("bind() error");if(listen(serv_sock, 5) == -1)error_handling("listen() error");//将serv_sock更改为非阻塞setnonblockingmode(serv_sock);//创建例程int epfd = epoll_create1(0);//为ep_events动态创建大小ep_events = malloc(sizeof(struct epoll_event)*EPOLL_SIZE);//注册服务端文件描述符event.events = EPOLLIN;event.data.fd = serv_sock;//为例程epfd注册文件描述符serv_sock,主要目的是监视serv_sock中的EPOLLIN事件epoll_ctl(epfd,EPOLL_CTL_ADD,serv_sock,&event);while(1) {//等待文件描述符发生变化,成功则返回发生事件的文件描述符数int event_cnt = epoll_wait(epfd,ep_events,EPOLL_SIZE,-1);if(event_cnt==-1) {puts("epoll_wait() error");break;}//遍历发生变化的文件描述符集合puts("return epoll wait");for(int i = 0;i < event_cnt;i++) {//如果发生变化的是服务端文件描述符,则接受连接if(ep_events[i].data.fd == serv_sock) {socklen_t adr_sz = sizeof(clnt_addr);int clnt_sock = accept(serv_sock,(struct sockaddr*)&clnt_addr,&adr_sz);//将clnt_sock更改为非阻塞setnonblockingmode(clnt_sock);//注册客户端文件描述符event.events = EPOLLIN|EPOLLET;event.data.fd = clnt_sock;epoll_ctl(epfd,EPOLL_CTL_ADD,clnt_sock,&event);printf("connected client:%d\n",clnt_sock);}//如果发生变化的是客户端文件描述符,收发消息else {//边缘触发方式中,发生事件时需要读取缓冲中所有数据,因此需要循环调用recv函数while(1) {memset(buf,0,BUF_SIZE);//有可能与多个客户端相连并且客户端同时发送消息,所以可能同时变换很多个文件描述符//i指的是当前发起请求的客户端int str_len = recv(ep_events[i].data.fd,buf,BUF_SIZE,0);printf("str_len=%d\n",str_len);if(str_len==0) {//将该文件描述符从例程中移除epoll_ctl(epfd,EPOLL_CTL_DEL,ep_events[i].data.fd,NULL);close(ep_events[i].data.fd);printf("closed client:%d\n",ep_events[i].data.fd);break;}//发生错误且errno值为EAGAIN时,说明没有数据可读else if(str_len<0){if(errno==EAGAIN) {printf("读取了全部数据\n");break;}}else {send(ep_events[i].data.fd,buf,str_len,0);//printf("发送了数据:%s\n",buf);  }}}}}close(serv_sock);close(epfd);free(buf);return 0;
}

注:条件触发回声服务器中,在 recv 接收数据时没有使用 while 循环读取数据。而在边缘触发方式中,发生事件时需要读取缓冲中所有数据,因此需要while循环读取缓冲区中的全部数据

15.2.5 边缘触发和条件触发孰优孰劣

边缘触发优点:可以分离接收数据和处理数据的时间点。
试想如果数据量很大,不能一次读取完,则需要产生很多次事件,服务端压力会很大。

条件触发和边缘触发的区别主要应该从服务器端实现模型的角度谈论,因此并不存在“谁更快?快多少?”的问题。

相关文章:

Linux网络编程(十五)——优于select的epoll

文章目录 15 优于select的epoll 15.1 epoll理解及应用 15.1.1 基于select的I/O复用技术速度慢的原因 15.1.2 select的优点 15.1.3 实现epoll时必要的函数和结构体 15.1.4 epoll_creat1 15.1.5 epoll_ctl 15.1.6 epoll_wait 15.1.7 基于epoll的回声服务器端 15.2 条件…...

PhotoShop学习07

1.为图像添加纹理 图层混合模式是混合 2 张图片的一种快捷方式&#xff0c;一般情况下为图片添加纹理外观可以用到混合模式。 这里有一副图片&#xff0c;我可以为其添加纹理&#xff0c;使之呈现出不同的效果。首先需要为当前图层添加一个纹理图片&#xff0c;可以使用置入嵌…...

【缓存击穿】Java的“SingleFlight”解决方案

在Java中实现类似Golang的SingleFlight机制&#xff0c;可以通过以下步骤解决缓存击穿问题。该方案使用ConcurrentHashMap管理并发请求&#xff0c;并通过CompletableFuture实现异步结果合并。 实现代码 import java.util.concurrent.Callable; import java.util.concurrent.…...

createContext+useContext+useReducer组合管理React复杂状态

createContext、useContext 和 useReducer 的组合是 React 中管理全局状态的一种常见模式。这种模式非常适合在不引入第三方状态管理库&#xff08;如 Redux&#xff09;的情况下&#xff0c;管理复杂的全局状态。 以下是一个经典的例子&#xff0c;展示如何使用 createContex…...

海外直播平台交互设计师简历模板

营销团队管理技巧培训PPT啊&#xff0c;其实是一个非常有用的工具呢&#xff01;它不仅能帮助管理者梳理思路&#xff0c;还能让团队成员快速掌握关键技能。说实话&#xff0c;一个好的PPT就像一位优秀的导师&#xff0c;在会议室里就能让人眼前一亮&#xff01;比如有一次我参…...

基于springboot微信小程序课堂签到及提问系统(源码+lw+部署文档+讲解),源码可白嫖!

摘要 随着信息时代的来临&#xff0c;过去的课堂签到及提问管理方式的缺点逐渐暴露&#xff0c;本次对过去的课堂签到及提问管理方式的缺点进行分析&#xff0c;采取计算机方式构建基于微信小程序的课堂签到及提问系统。本文通过阅读相关文献&#xff0c;研究国内外相关技术&a…...

MCU软件开发使用指针有哪些坑?

目录 1、空指针访问 2、野指针&#xff08;未初始化的指针&#xff09; 3、指针越界 4、内存泄漏 5、悬空指针 6、指针类型不匹配 7、多任务环境中的指针访问 8、对齐问题 在MCU软件开发中&#xff0c;使用指针虽然可以提高程序的灵活性和性能&#xff0c;但也存在许多…...

ubuntu 20.04 编译和运行SC-LeGo-LOAM

1.搭建文件目录和clone代码 mkdir -p SC-LeGo-LOAM/src cd SC-LeGo-LOAM/src git clone https://github.com/AbangLZU/SC-LeGO-LOAM.git cd .. 2.修改代码 需要注意的是原作者使用的是Ouster OS-64雷达&#xff0c;需要更改utility.h文件中适配自己的雷达类型&#xff0c;而…...

FPGA_DDR(一) 仿真

对ddr进行读写实验&#xff0c;用了vivado的ddr的模型进行仿真 1 创建AXI_mig的ip核 选择axi 选择自己的型号&#xff0c;这里是ddr的位宽32&#xff0c;但是axi的话是256位宽 选择nobuffer&#xff0c;没有缓冲器 选择自己匹配引脚 默认 后面默认即可 生成ip&#xff0c;时间…...

【Spec2MP:项目管理之项目人力管理】

芯片设计项目中如何打造战斗力强悍的团队&#xff1f; 引言&#xff1a;芯片设计项目的核心是人 芯片设计是一项高度复杂、跨学科协作的工程&#xff0c;团队的专业性、协作效率和凝聚力直接影响项目成败。本文结合某芯片项目人力管理文档&#xff0c;从目标、职责、价值观、架…...

windows10下PointNet官方代码Pytorch实现

PointNet模型运行 1.下载源码并安装环境 GitCode - 全球开发者的开源社区,开源代码托管平台GitCode是面向全球开发者的开源社区,包括原创博客,开源代码托管,代码协作,项目管理等。与开发者社区互动,提升您的研发效率和质量。https://gitcode.com/gh_mirrors/po/pointnet.pyto…...

阿里云大模型训练与推理开发

本文主要描述阿里云大模型开发环境的搭建、训练数据集的制作流程、大模型如何训练数据集以及如何利用已训练完成的模型执行推理。 开发环境搭建 ModelScope社区是阿里云通义千问开源的大模型开发者社区。 如上所示&#xff0c;安装ModelScope社区大模型基础库开发框架的命令行…...

图灵逆向——题一-动态数据采集

目录列表 过程分析代码实现 过程分析 第一题比较简单&#xff0c;直接抓包即可&#xff0c;没有任何反爬&#xff08;好像头都不用加。。。&#xff09; 代码实现 答案代码如下&#xff1a; """ -*- coding: utf-8 -*- File : .py author : 鲨鱼爱兜兜 T…...

GS069W电动工具直流调速电路深度解析

产品概述 GS069W是我们推出的CMOS专用调速集成电路&#xff0c;采用SOP8封装&#xff0c;内置15V稳压结构&#xff0c;具有宽电压输入&#xff08;4-24V&#xff09;、低功耗、强抗干扰等特点&#xff0c;专为电动工具调速设计。 核心参数 工作电压&#xff1a;4-24V&#xff…...

PyQt6实例_A股日数据维护工具_下载某个股票未复权数据

目录 前置&#xff1a; 相关代码&#xff1a; 1 工作类 2 数据库交互 3 主界面调用 视频 前置&#xff1a; 1 本系列将以 “PyQt6实例_A股日数据维护工具” 开头放置在“PyQt6实例”专栏 2 日数据可在“数据库”专栏&#xff0c;“PostgreSQL_”开头系列博文中获取 3 权…...

【蓝桥杯】算法笔记6

1. 可行性剪枝应用 1.1. 题目 题目描述: 给定一个正整数n和一个正整数目标值target,以及一个由不同正整数组成的数组nums。要求从nums中选出若干个数,每个数可以被选多次,使得这些数的和恰好等于target。问有多少种不同的组合方式? 输入: 第一行:n和target,表示数组…...

C++ 中日期类的输入输出操作符重载实践

目录 引言 预备知识 输出流操作符 operator<< 重载 为什么要返回 ostream& 输入流操作符 operator>> 重载 实现思路 测试代码 总结 引言 在 C 编程中&#xff0c;当我们自定义数据类型时&#xff0c;为了让其能像内置类型一样方便地进行输入输出操…...

图论:最小生成树

最小生成树 &#xff08;无向无环图&#xff09; 概念 1.Prim算法 P3366 【模板】最小生成树 - 洛谷 邻接矩阵实现 #include<iostream> #include<cstring> using namespace std; const int INF 0x3f3f3f3f; const int N 5e3 10; int dis[N]; //记录每个结点到…...

linux中CosyVoice声音克隆安装教程——TTS文本转语音(数字人组件)

CosyVoice 作为一款先进的语音合成解决方案&#xff0c;其设计理念在于提供高效、稳定且灵活的语音生成工具。本教程将从环境配置、依赖安装、模型下载到服务部署全流程进行详细介绍&#xff0c;旨在为用户提供前瞻性的技术指导&#xff0c;同时兼顾细节解析和专业名词解释&…...

智能手表该存什么音频和文本?场景化存储指南

文章目录 为什么需要“场景化存储”&#xff1f;智能手表的定位手机替代不了的场景碎片化的场景存储 音频篇&#xff1a;智能手表该存什么音乐和音频&#xff1f;运动场景通勤场景健康场景 文本篇&#xff1a;哪些文字信息值得放进手表&#xff1f;&#xff08;部分情况可使用图…...

怎么检查网站CDN缓存是否生效

为什么要使用CDN缓存&#xff1f; 网站使用缓存可显著提升加载速度&#xff0c;减少服务器负载和带宽消耗&#xff0c;优化用户体验&#xff0c;增强架构稳定性&#xff0c;助力SEO优化&#xff0c;实现资源高效利用与性能平衡。 通过合理配置 CDN 缓存策略&#xff0c;可降低…...

win10安装gitbash工具

问题描述:在Windows下没有预装bash命令处理工具 # WInR输入cmd回车进入命令行,执行以下命令出现乱码 bash 无法使用bash命令 解决方案&#xff1a;下载安装gitbash命令行工具 Git Bash 是一个在 Windows 上运行的终端仿真器&#xff0c;集成了 Git 和 Bash shell&#xff0…...

买不起了,iPhone 或涨价 40% ?

周知的原因&#xff0c;新关税对 iPhone 的打击&#xff0c;可以说非常严重。 根据 Rosenblatt Securities分析师的预测&#xff0c;若苹果完全把成本转移给消费者。 iPhone 16 标配版的价格&#xff0c;可能上涨43%。 iPhone 16 标配的价格是799美元&#xff0c;上涨43%&am…...

企业级 ClickHouse Docker 离线部署实践指南20250407

企业级 ClickHouse Docker 离线部署实践指南 引言 在数据分析与日志处理日益重要的今天&#xff0c;ClickHouse 凭借其高性能、列式存储架构&#xff0c;成为企业在大数据分析中的首选引擎之一。本文基于一位金融行业从业者在离线网络环境中部署 ClickHouse 的真实实践过程&a…...

多域名​ SSL 证书能保护多少个域名?

一、基础保护数量范围​ 多域名 SSL 证书&#xff0c;顾名思义&#xff0c;可保护多个不同域名。通常情况下&#xff0c;不同证书颁发机构&#xff08;CA&#xff09;设定的基础保护数量有所差异。一般的多域名 SSL 证书能保护2 至 5 个域名&#xff0c;这些域名可以是完全独立…...

Linux系统学习Day04 阻塞特性,文件状态及文件夹查询

知识点4【文件的阻塞特性】 文件描述符 默认为 阻塞 的 比如&#xff1a;我们读取文件数据的时候&#xff0c;如果文件缓冲区没有数据&#xff0c;就需要等待数据的到来&#xff0c;这就是阻塞 当然写入的时候&#xff0c;如果发现缓冲区是满的&#xff0c;也需要等待刷新缓…...

【AI】高效地使用 AI 模型的 Prompt(提示词)

明确任务和目标 在使用 Prompt 之前&#xff0c;要清楚知道自己想要通过 AI 模型完成什么任务&#xff0c;例如生成文本、回答问题、进行翻译或创作故事等。明确的目标有助于构建更有针对性的 Prompt&#xff0c;引导模型生成符合期望的结果。 精准描述问题 提供具体细节&am…...

第二十:mysql——Undo Log、Redo Log和Binlog

二进制日志binlog&#xff08;归档日志&#xff09;、 事务日志redo log&#xff08;重做日志&#xff09; MySQL实例挂了或者宕机了&#xff0c;重启的时候InnoDB存储引擎会使用rede log日志恢复数据&#xff0c;保证事务的持久性和完整性 和undo log&#xff08;回滚日志&a…...

LogicFlow-前端流程图开发

LogicFlow-前端流程图开发 一、安装使用 1、安装logicflow 通过npm安装logicflow npm install logicflow/core --save# 插件包&#xff08;不使用插件时不需要引入&#xff09; npm install logicflow/extension --save2、创建实例 import LogicFlow from "logicflow/…...

第四讲:类与对象(下)

目录 1、再谈构造函数 1.1、构造函数体赋值 1.2、初始化列表 1.3、explicit关键字 2、static成员 3、友元 3.1、友元函数 3.2、友元类 4、内部类 5、匿名对象 6、拷贝对象时的优化&#xff08;了解&#xff09; 7、重新理解类与对象 8、日期类的实现 9、练习题 9…...

ReAct 框架 | 提示词工程(1)

ReAct 框架 1、什么是 ReAct 框架&#xff1f;2、基于 ReAct 框架的提示词3、结合 LangChain 框架使用4、总结 1、什么是 ReAct 框架&#xff1f; ReAct &#xff1a; Reasoning Acting &#xff0c;将推理与外部工具调用结合&#xff0c;通过交互式探索解决复杂问题。 优点…...

第一部分——Docker篇 第一章 Docker容器

关于系统的改造探索 开篇&#xff1a;系统改造的调研报告 第一部分——Docker篇 第一章 Docker容器 第二章 Docker安装 第三章 构建自定义镜像 第四章 搭建镜像仓库 第五章 容器编排 第六章 容器监控 文章目录 关于系统的改造探索第一部分——Docker篇 前言一、就是你了——…...

ubuntu,react的学习(1)

在此目录下&#xff0c;开启命令行 /home/kt/react 如下操作 tkt4028:~/react$ npm create vitelatest task-manager -- --template react Need to install the following packages: create-vite6.3.1 Ok to proceed? (y) y> npx > cva task-manager --template react…...

AR 赋能儿童娱乐:剧本杀与寻宝小程序搭建秘籍​

在科技飞速发展的当下&#xff0c;儿童娱乐领域正经历着一场创新变革。AR&#xff08;增强现实&#xff09;技术的融入&#xff0c;为儿童剧本杀与寻宝游戏带来了前所未有的沉浸式体验。通过搭建专属小程序&#xff0c;孩子们能够在虚拟与现实交织的世界中开启奇幻冒险。接下来…...

2017年-全国大学生数学建模竞赛(CUMCM)试题速浏、分类及浅析

2017年-全国大学生数学建模竞赛(CUMCM)试题速浏、分类及浅析 全国大学生数学建模竞赛(China Undergraduate Mathematical Contest in Modeling)是国家教委高教司和中国工业与应用数学学会共同主办的面向全国大学生的群众性科技活动,目的在于激励学生学习数学的积极性,提高学…...

密码学基础——分组密码的运行模式

前面的文章中文我们已经知道了分组密码是一种对称密钥密码体制&#xff0c;其工作原理可以概括为将明文消息分割成固定长度的分组&#xff0c;然后对每个分组分别进行加密处理。 下面介绍分组密码的运行模式 1.电码本模式&#xff08;ECB&#xff09; 2.密码分组链接模式&…...

zk源码—2.通信协议和客户端原理一

大纲 1.ZooKeeper如何进行序列化 2.深入分析Jute的底层实现原理 3.ZooKeeper的网络通信协议详解 4.客户端的核心组件和初始化过程 5.客户端核心组件HostProvider 6.客户端核心组件ClientCnxn 7.客户端工作原理之会话创建过程 1.ZooKeeper如何进行序列化 (1)什么是序列化…...

【NLP】Transformer网络结构(2)

一、Transformer 整体架构 Transformer 由 Encoder 和 Decoder 堆叠组成&#xff0c;每个 Encoder/Decoder 层包含以下核心模块&#xff1a; Encoder 层&#xff1a;Multi-Head Self-Attention → Add & LayerNorm → Feed-Forward → Add & LayerNormDecoder 层&…...

【LeetCode77】组合

题目描述 给定区间 [1, n] 和一个整数 k&#xff0c;需要返回所有可能的 k 个数的组合。 思路 算法选择&#xff1a;回溯算法 回溯算法是一种试探性搜索方法&#xff0c;非常适合用来解决组合问题。基本思想是&#xff1a; 从数字 1 开始&#xff0c;逐步构建组合。当当前组…...

1631. 最小体力消耗路径

文章目录 题意思路代码 题意 题目链接 思路 搜索 代码 class Solution { public:int minimumEffortPath(vector<vector<int>>& heights) {int m heights.size();int n heights[0].size();int x_add[] {0, 0, 1, -1};int y_add[] {1, -1, 0, 0};if (m …...

时间复杂度和空间复杂度

&#x1f31f; 各位看官好&#xff0c;我是maomi_9526&#xff01; &#x1f30d; 种一棵树最好是十年前&#xff0c;其次是现在&#xff01; &#x1f680; 今天来学习C语言的相关知识。 &#x1f44d; 如果觉得这篇文章有帮助&#xff0c;欢迎您一键三连&#xff0c;分享给更…...

Python基于OpenCV和SVM实现中文车牌识别系统GUI界面

说明&#xff1a;这是一个系统实战项目&#xff0c;如需项目代码可以直接到文章最后关注获取。 项目背景 随着智能交通系统和智慧城市的发展&#xff0c;车牌识别技术在车辆管理、交通监控、停车场收费等领域发挥着重要作用。传统的车牌识别系统主要针对英文和数字的识别&…...

用AbortController取消事件绑定

视频教程 React - &#x1f914; Abort Controller 到底是什么神仙玩意&#xff1f;看完这个视频你就明白了&#xff01;&#x1f4a1;_哔哩哔哩_bilibili AbortController的好处之一是事件绑定的函数已无需具名函数,匿名函数也可以被取消事件绑定了 //该代码2秒后点击失效…...

4月7日随笔

晚饭塔斯汀 下了晚自习买了一瓶百香果rio 还有一块五毛钱的老酸奶&#xff0c;这个糖吃的时候是真开心呀 英语课互动感觉越来越少了&#xff0c;我甚至看了十分钟的小排球 解析几何和微积分都听不进去了。就算坐在第三排还是会走神。但是不知道为什么我刷视频和打游戏的时…...

Android使用声网SDK实现音视频互动(RTC)功能

一、前期准备 1、注册声网账号 声网官网 2、创建项目 拿到AppID&#xff0c;主要证书 二、代码部分 先上一下官方提供的demo地址&#xff1a; Agora-RTC-QuickStart: 此仓库包含 Agora RTC Native SDK 的QuickStart示例项目。 - Gitee.comhttps://gitee.com/agoraio-comm…...

【go】slice的浅拷贝和深拷贝

浅拷贝(Shallow Copy) 浅拷贝是指只复制切片本身的结构&#xff08;指针、长度和容量&#xff09;&#xff0c;而不复制底层数组的元素。 实现方式 直接赋值&#xff1a; slice1 : []int{1, 2, 3} slice2 : slice1 // 浅拷贝切片操作&#xff1a; slice1 : []int{1, 2, 3} s…...

哑铃图:让数据对比一目了然【Dumbbell Chart】

没错&#xff0c;当我祭出 “哑铃” 阵列&#xff0c;你当如何破解&#xff0c;哈哈哈哈…此时&#xff0c;你可以适当怀疑笔者的精神状态了。但话说回来&#xff0c;如果稍加想象&#xff0c;把上图竖起来&#xff0c;“大致” 就是我要分享的 “哑铃图” 了。&#x1f611; …...

Spring Boot 集成 MongoDB 时自动创建的核心 Bean 的详细说明及表格总结

以下是 Spring Boot 集成 MongoDB 时自动创建的核心 Bean 的详细说明及表格总结&#xff1a; 核心 Bean 列表及详细说明 1. MongoClient 类型&#xff1a;com.mongodb.client.MongoClient作用&#xff1a; MongoDB 客户端核心接口&#xff0c;负责与 MongoDB 服务器建立连接、…...

水产养殖水下监控无人机推荐-P200PRO

水产养殖水下监控无人机推荐 | 潜 鲛 P200 PRO&#xff1a;您的“水下管家”&#xff0c;养鱼增产、降本增效的终极利器&#xff01; ——上海 棕航电子 科技&#xff0c;用技术守护每一方鱼塘 一、水产养殖的痛点&#xff1a;看不见的水下&#xff0c;才是赚钱的关键 …...

数据结构与算法-数学-基础数学算法(筛质数,最大公约数,最小公倍数,质因数算法,快速幂,乘法逆元,欧拉函数)

一&#xff1a;筛质数&#xff1a; 1-埃氏筛法 该算法核心是从 2 开始&#xff0c;把每个质数的倍数标记为合数&#xff0c;时间复杂度约为 O(nloglogn)。 #include <iostream> #include <vector>u sing namespace std; const int N 1000010; bool st[N]; …...