简单服务器通信、IO多路复用(select、poll、epoll)以及reactor模式。

news2024/11/19 23:16:24

网络编程学习

  • 简单服务器通信
    • TCP三次握手和四次挥手
      • 三次握手(如下图)
        • 常见问题?
      • 四次挥手
    • client和server通信写法
      • server端
      • client端
    • 通信双方建立连接到断开连接的状态转换
    • 怎么应对多用户连接?
      • 缺点
  • IO多路复用
    • select
      • 优缺点
    • poll
      • poll写法和改进点
    • epoll(使用最多,重中之重)
      • epoll写法和改进点
      • LT模式和ET模式
        • **LT模式**
        • **ET模式**:
        • LT模式如何解决数据保存问题?
  • reactor模式
    • 针对IO处理的两种写法
  • 思考题
    • main函数如何被执行的?
    • time_wait怎么产生的?如果有大量的time_wait是为什么?
    • close_wait怎么产生的?如果有大量的close_wait是为什么?
    • epoll里是否使用了mmap?
    • epoll是否是线程安全的?

简单服务器通信

TCP三次握手和四次挥手

三次握手(如下图)

  1. 首先由通信双方的某一方(后面就用client了,其实server和client都是可以的)发起连接请求,即发送一个带SYN标志位的TCP数据包给server端。
  2. server端收到数据包后,除了SYN位以外,还会增加一个ACK确认位,然后发送给clinet端。
  3. client端收到数据包后,给server端回复一个带ACK确认位的数据包给服务端,到此,通信双方就算建立连接了。
常见问题?
  • 为什么是三次握手而不是两次?
    • 因为TCP连接是安全可靠的,三次握手的话会保证双方都能接受到对方的数据,第一次服务端收到客户端消息后可以确定客户端发送数据没问题,然后第二次服务端发送给客户端消息,客户端可以确定服务端的收发都没有问题,但是服务端不知道客户端接收是否成功,所以,客户端还需要发一个数据包告诉服务端,它的接收也没有问题,因此,这里是三次连接而不是两次。

四次挥手

  1. 第一次挥手,是主动断开连接的一方,发送带有FIN和ACK标志位的TCP包。
  2. 第二次是被动断开连接一方在收到数据包后,立马回应一个ACK标志位的TCP包,告诉对方我收到了对方的包,但是数据还没有处理完成,还有些工作要处理。
  3. 第三次依然是被动断开连接一方在数据处理完成之后,会给对方发送FIN和ACK标志位的TCP包,告诉对方我数据处理完了,你可以关闭连接了。
  4. 第四次挥手,是主动断开方发送ACK标志位的TCP包,然后被动断开方就关闭连接了。

client和server通信写法

server端

  • 服务端编码步骤:
    1. 调用socket函数,获取一个sockfd(本质上是一个文件描述符),int sockfd = socket(AF_INET, SOCK_STREAM, 0);
    2. 调用bind函数,将sockfd和ip、端口号绑定
      1. 创建一个sockaddr_in结构体变量,通过改结构体设置服务器的IP地址和端口号(这里要调用htonl和htons转换字节序)。
      2. 调用bind函数,绑定sockfd和ip、端口号;bind(sockfd, (struct sockaddr*)&server_addr, sizeof(struct sockaddr))
    3. 调用listen函数去监听sockfd上的请求。listen(sockfd, 8)
    4. 调用accept函数(默认为堵塞)去接收客户端请求。
      1. 创建两个变量,一个是sockaddr_in结构体变量,存储用来通信的客户端信息,一个是len代表结构体变量的大小。
      2. 调用accept函数,函数返回一个fd(可以用来读写数据的fd),int clientfd = accept(sockfd, (struct sockaddr*)&client_addr, &len);
    5. 第五步就是一个while循环,然后循环里调用recv和send函数去读写数据。
  • 整体代码如下:
       int sockfd = socket(AF_INET, SOCK_STREAM, 0);
       struct sockaddr_in server_addr;
       memset(&server_addr, 0, sizeof(struct sockaddr_in));//赋值为空
       server_addr.sin_family = AF_INET;//IPV4协议
       server_addr.sin_addr.s_addr = htonl(INADDR_ANY);//本地地址,如果有多个IP不知道设置为哪个就可以用这个参数。
       server_addr.sin_port = htons(2204);
       if (-1 == bind(sockfd, (struct sockaddr*)&server_addr, sizeof(struct sockaddr))) {
           perror("bind error");
           return -1;
       }
       if (-1 == listen(sockfd, 8)) {
           perror("listen error");
           return -1;
       }
       //accept
       struct sockaddr_in client_addr;
       int len = sizeof(struct sockaddr);
       //clientfd用于收发数据的
       int clientfd = accept(sockfd, (struct sockaddr*)&client_addr, &len);
       printf("clientfd: %d, serverfd: %d\n", clientfd, sockfd);
       while (1) {
           //receive
           char buf[128] = {0};
           int recv_len = recv(clientfd, buf, 128, 0);
           printf("accept len: %d", recv_len);
           if (recv_len == -1){
               perror("recv error");
               break;
           } else if (recv_len == 0) {
               break;
           } else {
               printf("recv success\n");
           }
           //send
           printf("send");
           send(clientfd, buf, recv_len, 0);
           printf("send success");
       }
    

client端

client写法步骤:
1. 调用socket函数,获取一个sockfd(本质上是一个文件描述符),int sockfd = socket(AF_INET, SOCK_STREAM, 0);
2. 调用connect函数建立连接(这一步完成三次握手)
1. 创建一个sockaddr_in结构体变量,通过改结构体设置服务器的IP地址和端口号(这里要调用htonl和htons转换字节序)。
2. 调用connect函数,如果成功返回那么就可以直接用sockfd进行通信了。
3. 使用sockfd进行数据读写。

  • 整体代码如下:
    uint32_t lfd = socket(AF_INET, SOCK_STREAM, 0);
    if (lfd == -1) {
        perror("create socket");
        exit(-1);
    }
    //connect 172.17.71.122
    uint32_t dst = 0;
    inet_pton(AF_INET, "8.141.4.79", &dst);
    struct sockaddr_in server_addr;
    server_addr.sin_family = AF_INET;//IPV4协议
    server_addr.sin_addr.s_addr = dst;
    server_addr.sin_port = htons(2204);
    if (connect(lfd, (struct sockaddr*)&server_addr, sizeof(server_addr)) == -1) {
        perror("connect");
        exit(-1);
    }
    while (1) {
        for (int i = 0; i < 20; ++i) {
            const char* str = std::to_string(i).c_str();
            int write_fd = write(lfd, str, sizeof(str));
            char receive[1024] = "";
            int len1 = read(lfd, receive, sizeof(receive));
            if (len1 < 0) {
                perror("client read");
                exit(-1);
            } else if (len1 == 0) {
                printf("server closed!!!");
                break;
            } else {
                printf(" client read success: %s\n", receive);
            } 
            sleep(1);
        }
        sleep(2);
    }

通信双方建立连接到断开连接的状态转换

在这里插入图片描述

  • 建立连接:
    • 服务端:
      1. 初始状态为closed状态,调用listen函数后进入LISTEN状态。
      2. 收到SYN标志位的TCP包,并向对方发送SYN和ACK标志位的TCP包后,变为SYN_RCVD状态。
      3. 收到对方带ACK标志位的包后变为ESTABLISTED状态,至此连接建立。(这里不是accept去完成三次握手的,三次握手由TCP协议栈完成,accept只是堵塞获取客户端信息的)。
    • 客户端:
      1. 初始状态为closed状态,调用connect函数,发送SYN标志位的TCP包后进入SYN_SENT状态。
      2. 由TCP协议栈完成后续的握手环节,当connect函数成功返回后,直接进入ESTABLISTED状态。
  • 断开连接:断开连接一般情况下可以是任意一方发起,这里以客户端主动断开为例
    • 服务端:
      1. 当客户端主动断开连接后,服务端收到FIN位的TCP包,此时服务端的协议栈会立刻发送给对方一个ACK确认位,此时服务端酒进入了CLOSED_WAIT状态。
      2. 此时recv函数会继续去处理数据,当读到数据长度为0,并且数据全部处理完后,服务端主动调用close函数关闭通信fd,这时服务端会给客户端发送FIN标志位的TCP包,然后服务端进入LAST_ACK状态。
      3. 当服务端收到客户端的ACK标志位的包后,服务端由LAST_ACK转变为CLOSED状态。
    • 客户端:
      1. 客户端主动调用closed函数关闭fd,那么TCP协议栈会给服务端发送一个带SYN标志位的TCP包,此后客户端进入FIN_WAIT1状态。
      2. 当收到服务端带ACK标志位的TCP包后,由FIN_WAIT1转变为FIN_WAIT2状态。
      3. 当收到服务端带FIN标志位的TCP包,并向服务端发送ACK标志位的TCP包后就进入了TIME_WAIT状态,该状态会持续2MSL,(MSL为网络包在网络中的最大存活时间。),当2MSL到达后,转换为CLOSED状态。
  • 注意:
    • TCP的状态转换是由TCP协议栈管理的,不是由代码管理。

怎么应对多用户连接?

  • 上面的简单通信过程只能一次处理一个用户,我们如果想要处理多用户连接可以采用多线程的方式。
  • 方法也比较简单,就是将IO操作以及业务处理逻辑放到一个函数里,然后accept只要返回了,那么就创建一个线程去执行回调函数。

缺点

  1. 当并发量特别大的时候,服务器会受到内存的影响,性能并不高,无法适应高并发场景。
  2. 为此,我们需要学习下面的IO多路复用,采用一个线程去处理多个请求。

IO多路复用

  • 简单来讲就是可以用一个线程去同时监听多个请求,并一起返回,同时处理,减少线程资源消耗。

select

  • select相当于一个代理,去帮我们检测有哪些fd有事件发生,然后返回给我们一个数组,我们需要遍历数组依次处理每个fd。

  • 实现步骤如下:

    1. 定义一个fd_set变量和maxfd变量。
      1. fd_set变量是一个结构体变量,内部存储了一个fd数组,然后我们调用FD_SET函数将sockfd添加到fd_set里。
      2. maxfd变量存储当前的最大fd值,初始值为sockfd,后续会及时更新。
    2. 调用select函数去监听事件。int ret_code = select(maxfd + 1, &r_set, NULL, NULL, NULL); 这里maxfd+1的目的是为了能够处理maxfd,因为select的实现比较老了。
    3. select函数返回后,我们需要写一个状态机,去判断是sockfd还是其他fd也有。两种不同的fd,会有不同的IO操作。
  • 代码如下:

    fd_set rf_set, r_set;
    FD_ZERO(&rf_set);//清空集合
    FD_SET(sockfd, &rf_set);//增加sockfd到集合
    
    int maxfd = sockfd;//设置最大fd
    while (1) {
        r_set = rf_set;
        int ret_code = select(maxfd + 1, &r_set, NULL, NULL, NULL);
    
        if (FD_ISSET(sockfd, &r_set)) {
            //accept
            struct sockaddr_in client_addr;
            int len = sizeof(struct sockaddr);
            int client_fd = accept(sockfd, (struct sockaddr*)&client_addr, &len);
            FD_SET(client_fd, &rf_set);//增加新的fd到fd_set里
            maxfd = maxfd > client_fd ? maxfd : client_fd;//更新maxfd
        } else {
            for (int i = sockfd + 1; i < maxfd + 1; ++i) {
                if (FD_ISSET(i, &r_set)) {
                    //receive
                    char buf[128] = {0};
                    int recv_len = recv(i, buf, 128, 0);
                    printf("accept len: %d", recv_len);
                    if (recv_len == -1){
                        perror("recv error");
                        break;
                    } else if (recv_len == 0) {
                        FD_CLR(i, &rf_set);
                        close(i);//记得close,不然会一直在close_wait状态。
                        break;
                    } else {
                        printf("recv success\n");
                    }
                    //send
                    printf("send");
                    send(i, buf, recv_len, 0);
                    printf("send success");
                }
            }
        }
    }
    

优缺点

  • 优点:相较于多线程实现法,进行了极大的优化,减少线程资源消耗。
  • 缺点:
    1. 函数参数太多了,一共有五个参数,中间三个参数分别为r_set、w_set、e_set分别对应不同的事件。
    2. 使用select会频繁的进行内存拷贝,每当处理完数据就会将fd_set拷贝到内核态,而每次有事件发生都会将fd_set拷贝到用户态,频繁拷贝,浪费很多资源,性能很低。
    3. select能够处理的最大连接数为1024个。

poll

poll写法和改进点

  • 改进点:

    1. 优化了函数参数,将中间三个参数变为一个,增加了一个结构体,用来存储监听哪些事件(读写)、是否监听以及fd。用户定义一个数组,作为传出参数,传给poll函数,等函数返回后,用户需要遍历数组,对于数组每个元素,如果revents被改变,那么就是发生的对应的事件,用户可以进行相应的操作。
    2. 因为用户可以自定义数组大小,所以poll解决了select只能监听1024个文件描述符的问题。
  • 缺点:

    • 底层的内核监听过程和select类似,也需要把数组从用户态拷贝到内核态,只是把修改fdset换成了每个数组元素中revents的值,其他相差不大。
  • 写法代码如下:

    // struct pollfd {
    //     int fd;			/* File descriptor to poll.  */
    //     short int events;		/* Types of events poller cares about.  */
    //     short int revents;		/* Types of events that actually occurred.  */
    // };
    struct pollfd fds[1024];
    fds[sockfd].fd = sockfd;
    fds[sockfd].events = POLLIN;
    int maxfd = sockfd;
    while (1) {
        int ret_code = poll(fds, maxfd + 1, -1);
    
        if (fds[sockfd].revents & POLLIN) {
            struct sockaddr_in client_addr;
            int len = sizeof(struct sockaddr);
            int client_fd = accept(sockfd, (struct sockaddr*)&client_addr, &len);
            fds[client_fd].fd = client_fd;
            fds[client_fd].events = POLLIN;
            maxfd = maxfd > client_fd ? maxfd : client_fd;
        } 
        for (int i = sockfd + 1; i < maxfd + 1; ++i) {
            if (fds[i].revents & POLLIN) {
                //receive
                char buf[128] = {0};
                int recv_len = recv(i, buf, 128, 0);
                printf("accept len: %d", recv_len);
                if (recv_len == -1){
                    perror("recv error");
                    break;
                } else if (recv_len == 0) {
                    fds[i].events = 0;
                    fds[i].fd = -1;
                    close(i);
                    break;
                } else {
                    printf("recv success\n");
                }
                //send
                printf("send");
                send(i, buf, recv_len, 0);
                printf("send success");
            }
        }
    }
    

epoll(使用最多,重中之重)

epoll写法和改进点

  • 改进点:

    1. epoll是在内核中申请一块缓存,存放一个节点,里面包含一个红黑树和双向链表,采用红黑树存储要监听的fd,然后将监听到的fd存储到一个双向链表里,这样加大了内核的查询效率,以及用户拿到的是被触发的结果集,不需要再去遍历所有的fd了,也就是不需要用户自己维护maxfd了。
    2. 虽然epoll也定义了一个数组,但是epoll的写法采用的共享内存的方式,而select和poll是单纯的拷贝。
  • 缺点:

    • 当服务遇到大量的短连接时,就会频繁的调用epoll_ctl系统调用,消耗系统资源。
  • 写法步骤:

    1. 调用epoll_create系统调用会在内核为该进程创建一个句柄,其中包含红黑树根节点和一个双向链表分别存储需要监听的节点和准备就绪的fd。
    2. 调用epoll_ctl系统调用,向红黑树节点中添加、删除、修改想要操作的fd,起初肯定是sockfd,并且注册一个回调函数,告诉内核如果有准备就绪的节点就添加到双向链表中。
    3. 调用epoll_wait系统调用,由内核去检测双向链表是否为空,如果为空,就sleep,直到链表里有数据时,epoll_wait被唤醒,将数据返回给用户,当然用户也可以自己设置超时时间,在该时间到达后,即使没有数据也会返回告知用户。
  • 代码如下:

    /*struct epoll_event {
            uint32_t events;	
            epoll_data_t data;	
        } __EPOLL_PACKED;
        */
        int epoll_fd = epoll_create(1);//这里只要大于0就可以,内部实现改为链表了。
    
        struct epoll_event ev;
        ev.events = EPOLLIN;
        ev.data.fd = sockfd;
        epoll_ctl(epoll_fd, EPOLL_CTL_ADD, sockfd, &ev);
    
        struct epoll_event epoll_events[1024] = {};
    
        while (1) {
            int ret_code = epoll_wait(epoll_fd, epoll_events, 1024, -1);
    
            for (int i = 0; i < ret_code; ++i) {
                int connt_fd = epoll_events[i].data.fd;
                if (connt_fd == sockfd) {
                    struct sockaddr_in client_addr;
                    int len = sizeof(struct sockaddr);
                    int client_fd = accept(sockfd, (struct sockaddr*)&client_addr, &len);
                    ev.events = EPOLLIN;
                    ev.data.fd = client_fd;
                    epoll_ctl(epoll_fd, EPOLL_CTL_ADD, client_fd, &ev);
                    printf("clientfd: %d\n", client_fd);
                } else if (epoll_events[i].events & EPOLLIN){
                    //receive
                    char buf[10] = {0};
                    int recv_len = recv(connt_fd, buf, 10, 0);
                    printf("accept len: %d", recv_len);
                    if (recv_len == -1){
                        perror("recv error");
                        close(connt_fd);
                        continue;
                    } else if (recv_len == 0) {
                        epoll_ctl(epoll_fd, EPOLL_CTL_DEL, epoll_events[i].data.fd, NULL);
                        close(i);
                        continue;
                    } else {
                        printf("recv success\n");
                    }
                    //send
                    printf("send");
                    send(connt_fd, buf, recv_len, 0);
                    printf("send success");
                }
            }
        }
    

LT模式和ET模式

LT模式
  • 是什么?
    • 我们在处理数据的时候,会有一个存储数据的buffer数组,如果buffer数组的长度小于要读取的数据长度,那么recv函数只会读取buffer长度的数据,然后等下次epoll_wait返回后,再继续读取数据,但是上次的数据需要保存,不然就丢失了。
  • 应用场景:
    • 可以采用LT模式解决粘包问题,先读取数据的长度,然后循环把数据全部读取出来。
ET模式
  • 是什么?
    • 内核针对每一个fd只返回一次(也就是说针对每次事件,epoll_wait只返回一次),后续就不再对该fd进行返回,这就使得用户必须一次性读完,否则只能等到下次fd有事件触发的时候才能接着读取上次的数据。
  • 应用场景:
    • 当处理大文件的时候,一次性读取所有数据。因为如果采用LT模式的话,文件太大会针对这个fd频繁调用epoll_wait系统调用。
LT模式如何解决数据保存问题?
  • 直接看代码(定义一个全局变量,存储每个fd对应的buffer,并且记录上次读到的位置idx):
    struct fd_events    
    {
        int fd;
        char r_buffer[128];
        char w_buffer[128];
        int r_idx;
        //这里还可以添加事件以及回调函数。
    };
    
    //解决LT模式下无法存数据的问题,每次读完后拼接到buffer后,并修改idx到数组末尾
    struct fd_events all_events[1024] = {};
    

reactor模式

针对IO处理的两种写法

  • 针对IO状态机处理
    • 上面的epoll示例就是针对fd去处理的,对sockfd和其他fd分别处理。
  • 针对事件状态机划分
    • 因为epoll_event里的events是有不同的事件的,我们可以针对不同的事件进行状态机判断,然后分别调用不同的回调函数。
    • 代码如下:
         if (epoll_events[i].events & EPOLLIN) {
             
         } else if (epoll_events[i].events & EPOLLOUT) {
      
         } else if (epoll_events[i].events & EPOLLERR) {
      
         } else {
      
         }
      
  • reactor模式就是针对不同事件对应不同的回调函数的模式。

思考题

main函数如何被执行的?

time_wait怎么产生的?如果有大量的time_wait是为什么?

close_wait怎么产生的?如果有大量的close_wait是为什么?

epoll里是否使用了mmap?

epoll是否是线程安全的?

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

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

相关文章

算法---动态规划练习-5(下降路径最小和)

下降路径最小和 1. 题目解析2. 讲解算法原理方法一方法二 3. 编写代码法一法二 1. 题目解析 题目地址&#xff1a;点这里 2. 讲解算法原理 方法一 首先&#xff0c;通过matrix的大小确定矩阵的行数m和列数n。 创建一个大小为(m1) (n2)的二维动态规划数组dp&#xff0c;其中d…

OC 技术 苹果内购

一直觉得自己写的不是技术&#xff0c;而是情怀&#xff0c;一个个的教程是自己这一路走来的痕迹。靠专业技能的成功是最具可复制性的&#xff0c;希望我的这条路能让你们少走弯路&#xff0c;希望我能帮你们抹去知识的蒙尘&#xff0c;希望我能帮你们理清知识的脉络&#xff0…

2022 年甘肃省职业院校技能大赛 高职组 网络系统管理竞赛 网络构建模块试题

2022 年甘肃省职业院校技能大赛 高职组网络系统管理竞赛 网络构建模块试题 目 录 考试说明… 3 任务描述… 3 任务清单… 3 &#xff08;一&#xff09;基础配置… 3 &#xff08;二&#xff09;有线网络配置… 4 &#xff08;三&#xff09;无线网络配置… 6 &#xff08;四&a…

惊艳亮相!新版飞桨星河社区邀您共赴星辰大海

亲爱的开发者们&#xff1a; 一场始于好奇、归于热爱的邂逅&#xff0c;让开发者们在飞桨星河社区相聚&#xff0c;相逢于活动、课程、比赛、项目……五年来&#xff0c;千万开发者的共同参与&#xff0c;让这个社区更富活力、更加温暖。 2018年&#xff0c;AI Studio 1.0版本上…

RHCE- 4-Web服务器(2)

基于https协议的静态网站 概念解释 超文本传输协议HTTP协议被用于在Web浏览器和网站服务器之间传递信息。 HTTP协议以明文方式发送内容&#xff0c;不提供任何方式的数据加密&#xff0c;如果攻击者截取了Web浏览器和网站服务器之间的传输报文&#xff0c;就可以直接读懂其中…

LVS负载均衡(load balance)

一 LVS LVS&#xff1a;Linux Virtaul Server&#xff0c;该软件的功能是实现 LB&#xff08;load balance&#xff09; 二LVS 的三种工作模式 1.NAT 模式&#xff08;NAT&#xff09; LVS 服务器同时充当一台 NAT 网关&#xff0c;拥有公有 IP &#xff0c;同时负责将针对此…

Apple Vision Pro应用合集

这里给大家分享一个网站&#xff0c;手机了最新的apple vision pro 上面运行的应用。 1、查找应用&#xff1a;用户可以浏览特色推荐的应用&#xff0c;或者通过随机挑选功能发现新的应用。 2、社区交流&#xff1a;提供社区功能&#xff0c;用户可以在这里交流使用体验、分享…

Chrome/Edge 使用 Markdown Viewer 查看 Markdown 格式文件

Chrome/Edge 使用 Markdown Viewer 查看 Markdown 格式文件 0. 引言1. 安装 Markdown Viewer 插件2. 使用 Markdown Viewer 阅读 Markdown 格式文件 0. 引言 大部分程序员都喜欢 Markdown 格式的文件&#xff0c;这时给一些没有在电脑上安装 Markdown 编辑器的同事分享资料时&…

SAP gui 组服务器 提示 Error service sapmsPRD unknown

/etc/hosts 追加IP地址和域名的配对关系 /etc/services 追加 sapms[sid] 3601/tcp

java数据结构与算法刷题-----LeetCode665. 非递减数列

java数据结构与算法刷题目录&#xff08;剑指Offer、LeetCode、ACM&#xff09;-----主目录-----持续更新(进不去说明我没写完)&#xff1a;https://blog.csdn.net/grd_java/article/details/123063846 文章目录 解题思路&#xff1a;时间复杂度O( n n n)&#xff0c;空间复杂度…

在vscode终端terminal加大栈空间How to increase max stack size in c++ using vscode

参考:https://devpress.csdn.net/cloud/63268204fd0b112779162383.html Answer a question In many dynamic programming and graph problems it is required to do long depth recursion. I am currently using vscode and mingw in windows for my c programs. But in defau…

计算机视觉之三维重建(4)---三维重建基础与极几何

文章目录 一、三维重建基础1.1 问题引入1.2 线性解法1.3 非线性解法1.4 多视图几何的关键问题 二、极几何与基础矩阵2.1 极几何2.2 极几何特例2.3 本质矩阵2.4 本质矩阵的性质2.5 基础矩阵2.6 基础矩阵的性质 三、基础矩阵估计 一、三维重建基础 1.1 问题引入 1. 从单张图像恢…

MySQL Explain 字段详解

Explain 工具介绍 Explain 一般被称为解释器&#xff0c;通过 Explain 工具&#xff0c;我们能分析我们使用的查询语句或是结构的性能瓶颈&#xff0c;它提供 MySQL 如何执行语句的信息。 使用语法&#xff1a; explain [extended|partition] select在 select 关键字前加 ex…

【多模态融合】SuperFusion 激光雷达与相机多层次融合 远距离高清地图预测 ICRA 2024

前言 本文介绍激光雷达与相机进行多层次融合&#xff0c;包括数据级融合、特征级融合和BEV级融合。 融合后的BEV特征可以支持不同的任务头&#xff0c;包括语义分割、实例编码和方向预测&#xff0c;最后进行后处理生成高清地图预测&#xff0c;它是来自ICRA 2024的。 会讲解…

【C++ leetcode】双指针(专题完结)

15. 三数之和 题目 给你一个整数数组 nums &#xff0c;判断是否存在三元组 [nums[i], nums[j], nums[k]] 满足 i ! j、i ! k 且 j ! k &#xff0c;同时还满足 nums[i] nums[j] nums[k] 0 。请 你返回所有和为 0 且不重复的三元组。 注意&#xff1a;答案中不可以包含重复的…

3.26学习总结

java 实例变量和局部变量 实例变量是记录这个类中对象的特点的每一个对象的实例变量都可以不同(例如名字,性别等),其中一个对象的实例变量改变不会影响其他的变量. 类变量是一种特殊的实例变量,他的特殊在于所有的对象的类变量都是相同的,当一个对象改变了类变量那么所有对象…

0101支付安全-支付模块-项目实战

文章目录 一、信息安全的基础-机密性1 相关概念2 对称加密和非对称加密 二、身份认证三 摘要算法四、数字签名五、数字证书结语 在支付过程中&#xff0c;设计多方的敏感信息&#xff0c;那么安全尤为重要。下面先简单介绍下&#xff0c;相关概念。 一、信息安全的基础-机密性 …

Java项目:76 Springboot学生读书笔记共享

作者主页&#xff1a;源码空间codegym 简介&#xff1a;Java领域优质创作者、Java项目、学习资料、技术互助 文中获取源码 项目介绍 本文从管理员、用户的功能要求出发&#xff0c; 读书笔记共享平台系统中的功能模块主要是实现管理员&#xff1b;首页、个人中心、用户管理、…

Intellij IDEA构建Android开发环境

Intellij IDEA创建项目时没有Android的选项 进设置&#xff08;Intellij IDEA - Settings - Plugins &#xff09; 再次创建项目可以看到Android的选项 解决Android导入项目时Gradle下载速度慢/超时/失败

Ansys Speos | Light Expert Group探测器组使用技巧

附件下载 联系工作人员获取附件 概述 相机挡板的设计需要在光路的不同位置同步多个照度图&#xff0c;以尽量减少杂散光。2023R2 Speos提供了一种新的探测器&#xff0c;用于高阶杂散光分析&#xff0c;可以同时对多个探测器进行光线追迹。Light Expert工具可以即时过滤3D视…