IO多路复用-select的使用详解【C语言】

news2024/11/25 3:04:56

1.多进程/线程并发和IO多路复用的对比

IO多路转接也称为IO多路复用,它是一种网络通信的手段(机制),通过这种方式可以同时监测多个文件描述符并且这个过程是阻塞的,一旦检测到有文件描述符就绪( 可以读数据或者可以写数据)程序的阻塞就会被解除,之后就可以基于这些(一个或多个)就绪的文件描述符进行通信了。通过这种方式在单线程/进程的场景下也可以在服务器端实现并发。常见的IO多路转接方式有:select、poll、epoll。

下面先对多线程/多进程并发和IO多路转接的并发处理流程进行对比(服务器端):

多线程/多进程并发

  1. 主线程/父进程

    • 调用accept()阻塞等待客户端连接请求。
    • 当有新的连接请求到达时,解除阻塞,建立连接并创建一个新的子线程或子进程来处理该连接。
  2. 子线程/子进程

    • 与客户端进行通信,使用read() / recv()来接收数据,如果没有数据则阻塞,直到数据到达。
    • 使用write() / send()来发送数据,如果写缓冲区满了则阻塞,直到可以写入数据。
  3. 优点

    • 简单直观,编程模型类似于顺序执行。
    • 可以利用多核处理器的优势,实现真正的并行处理。
  4. 缺点

    • 每个连接需要一个单独的线程或进程,创建和销毁线程/进程的开销较大。
    • 系统资源消耗较高,特别是在面对大量连接时,上下文切换开销增加。

IO多路复用并发

  1. IO多路复用

    • 使用selectpollepoll等函数委托操作系统监视多个文件描述符(包括监听和通信的文件描述符)。
    • 调用这些函数会阻塞进程,直到有文件描述符准备好IO操作(连接请求到达或有数据可读写)。
  2. 监听文件描述符

    • 一旦有新的连接请求到达,不会阻塞程序,因为select等函数已经通知监听文件描述符就绪。
    • 可以立即调用accept()接受连接,而不用等待。
  3. 通信文件描述符

    • 一旦有数据可读或可写,对应的通信操作不会阻塞,可以直接进行读写操作。
    • 不同于多线程/多进程模型,避免了为每个连接创建线程或进程的开销,大大减少了系统资源消耗。
  4. 优点

    • 系统开销小,不必频繁创建和销毁线程/进程。
    • 可以处理大量连接而不受资源限制的影响,效率较高。
  5. 缺点

    • 编程复杂度较高,因为需要管理和维护多个文件描述符的状态。
    • 在某些情况下,如文件描述符数量非常大时,性能可能略逊于epoll等更高级的IO复用机制。

2.select的使用

在IO多路复用中,select是最基础和最古老的一种实现方式之一。select函数允许程序同时监视多个文件描述符(sockets、stdin、stdout等),并在其中任何一个文件描述符准备好进行IO操作(如读或写)时通知程序。

程序猿通过调用这个函数可以委托内核帮助我们检测若干个文件描述符的状态,其实就是检测这些文件描述符对应的读写缓冲区的状态:

  • 读缓冲区:检测里边有没有数据,如果有数据该缓冲区对应的文件描述符就绪
  • 写缓冲区:检测写缓冲区是否可以写(有没有容量),如果有容量可以写,缓冲区对应的文件描述符就绪
  • 读写异常:检测读写缓冲区是否有异常,如果有该缓冲区对应的文件描述符就绪

委托检测的文件描述符被遍历检测完毕之后,已就绪的这些满足条件的文件描述符会通过select()的参数分3个集合传出,程序猿得到这几个集合之后就可以分情况依次处理了。

2.1select函数

select函数是一个用于IO多路复用的系统调用,允许程序监视多个文件描述符(sockets、stdin、stdout等),并在其中任何一个文件描述符准备好进行IO操作(如读或写)时通知程序。

#include <sys/select.h>
struct timeval {
    time_t      tv_sec;         /* seconds */
    suseconds_t tv_usec;        /* microseconds */
};
int select(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout);
参数说明
    nfds:监视的所有文件描述符中最大的文件描述符加1。通常可以通过计算出最大的文件描述符值加1来设置这个参数,例如 (fd + 1),其中 fd 是最大的文件描述符。内核需要线性遍历这些集合中的文件描述
符,这个值是循环结束的条件
    readfds:要监视读操作的文件描述符集合。传入传出参数,读集合一般情况下都是需要检测的,这样
才知道通过哪个文件描述符接收数据
    writefds:要监视写操作的文件描述符集合。传入传出参数,如果不需要使用这个参数可以指定为NULL
    exceptfds:要监视异常情况的文件描述符集合(一般设置为 NULL,表示不关心异常情况)。传入传出
参数,如果不需要使用这个参数可以指定为NULL
    timeout:超时时间,用于设置select函数的阻塞时间。它可以指定程序阻塞的最长时间。如果设置为 NULL,select将一直阻塞,直到有事件发生;如果设置为指向具有零时间的 timeval 结构的指针,则
select将立即返回;如果 timeout 不为 NULL,并且 select 在指定时间内没有检测到事件,则返回超时。
返回值说明
    -1:如果 select 函数调用失败,通常是由于参数错误或者系统调用中断(如信号中断)。
    0:如果 select 超时,即在指定的超时时间内没有文件描述符准备好进行IO操作。
    大于 0:返回已经准备好进行IO操作的文件描述符的数量。在返回时,fd_set 集合中会被修改,标识
出哪些文件描述符已经准备好。

注意事项和细节

  • 文件描述符集合的操作

    • 在调用 select 之前,需要使用 FD_ZEROFD_SET 等函数来准备好 readfdswritefdsexceptfds 这些文件描述符集合。
  • 文件描述符数量限制

    • 某些系统对单个进程能够监视的文件描述符数量有限制,可以通过 getrlimit() 函数查询当前进程的文件描述符数量限制。
  • 超时设置

    • 如果不需要超时等待,可以将 timeout 参数设置为 NULL,此时 select 函数将一直阻塞,直到有文件描述符准备好。
  • 并发性和效率

    • select 函数的效率可能会随着文件描述符数量的增加而下降,因为它在内核中需要线性扫描整个文件描述符集合。
  • 可移植性

    • select 函数几乎在所有主流操作系统上都有实现,因此具有较好的跨平台性。

另外初始化fd_set类型的参数还需要使用相关的一些列操作函数,具体如下:

1. void FD_CLR(int fd, fd_set *set);
功能:从文件描述符集合中删除指定的文件描述符。
参数:
fd:要从集合中删除的文件描述符。
set:要操作的文件描述符集合指针。
作用:将文件描述符集合 set 中的 fd 对应的标志位清零,即将其从集合中移除。
2. int FD_ISSET(int fd, fd_set *set);
功能:检查文件描述符集合中特定文件描述符的状态。
参数:
fd:要检查的文件描述符。
set:要操作的文件描述符集合指针。
返回值:
如果文件描述符 fd 在 set 集合中被设置(即处于就绪状态),返回非零值;
否则返回0。
作用:通常用于在调用 select 函数后,检查哪些文件描述符已经准备好进行IO操作。
3. void FD_SET(int fd, fd_set *set);
功能:向文件描述符集合中添加指定的文件描述符。
参数:
fd:要添加的文件描述符。
set:要操作的文件描述符集合指针。
作用:将文件描述符集合 set 中的 fd 对应的标志位设置为1,表示将该文件描述符加入到监视列表中。
4. void FD_ZERO(fd_set *set);
功能:清空文件描述符集合。
参数:
set:要操作的文件描述符集合指针。
作用:将文件描述符集合 set 中的所有标志位清零,即清除所有文件描述符的状态,使集合为空集。

2.2细节说明

在select()函数中第2、3、4个参数都是fd_set类型,它表示一个文件描述符的集合,类似于信号集 sigset_t,这个类型的数据有128个字节,也就是1024个标志位,和内核中文件描述符表中的文件描述符个数是一样的。

sizeof(fd_set) = 128 字节 * 8 = 1024 bit      // int [32]

这并不是巧合,而是故意为之。这块内存中的每一个bit 和 文件描述符表中的每一个文件描述符是一一对应的关系,这样就可以使用最小的存储空间将要表达的意思描述出来了。

下图中的fd_set中存储了要委托内核检测读缓冲区的文件描述符集合。

  • 如果集合中的标志位为0代表不检测这个文件描述符状态
  • 如果集合中的标志位为1代表检测这个文件描述符状态

内核在遍历这个读集合的过程中,如果被检测的文件描述符对应的读缓冲区中没有数据,内核将修改这个文件描述符在读集合fd_set中对应的标志位,改为0,如果有数据那么这个标志位的值不变,还是1。

当select()函数解除阻塞之后,被内核修改过的读集合通过参数传出,此时集合中只要标志位的值为1,那么它对应的文件描述符肯定是就绪的,我们就可以基于这个文件描述符和客户端建立新连接或者通信了。

2.3select处理流程

  • 初始化准备

    • 创建监听的套接字 lfd,并设置为非阻塞模式(可选但推荐,可以通过 fcntl 函数实现)。
    • 绑定套接字到本地的IP和端口,使用 bind() 函数。
    • 开始监听连接请求,使用 listen() 函数。
  • 文件描述符集合准备

    • 创建一个 fd_set 结构体对象,用于存储所有需要监视的文件描述符。
    • 使用 FD_ZERO() 初始化该结构体,清空所有文件描述符的状态。
    • 使用 FD_SET() 将监听套接字 lfd 添加到监视集合中。
  • 循环处理事件

    • 进入主循环,周期性地调用 select() 函数来等待文件描述符的就绪事件。
  • 调用 select() 函数

    • 使用 select() 函数阻塞等待事件发生。一旦有文件描述符就绪(可以读或写),select() 函数将返回。
  • 处理就绪事件

    • 使用 FD_ISSET() 函数遍历文件描述符集合,检查哪些文件描述符已经就绪。
    • 如果监听套接字 lfd 就绪,表示有新的客户端连接请求到来:
      • 使用 accept() 函数接受新连接,并创建新的通信套接字 cfd
      • 将新的通信套接字 cfd 添加到文件描述符集合中,使用 FD_SET()
      • 如果 cfd 大于当前的 max_fd,更新 max_fd
      • 输出连接的客户端的 IP、端口和文件描述符信息。
    • 如果是其他通信套接字 cfd 就绪,表示有数据到达或可以发送:
      • 调用相应的读取和写入函数处理客户端的数据收发操作。
      • 如果读取返回0,表示客户端关闭连接,应该调用 close() 关闭套接字,并使用 FD_CLR() 将其从文件描述符集合中移除。
      • 如果写入遇到缓冲区满的情况,可以选择继续尝试写入或者延迟处理。
  • 错误处理和异常情况

    • 在调用 select()accept()read()write() 等函数时,需要适时处理可能的错误情况和异常。
    • 对于非阻塞套接字,当 accept() 返回 EAGAINEWOULDBLOCK 错误时,应适当延迟重试或放入队列等待处理。
  • 循环迭代

    • 回到第3步,继续等待和处理下一轮的事件。

示例代码:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <arpa/inet.h>
#include <errno.h>

#define MAX_CLIENTS 10
#define PORT 9999

int main() {
    int lfd, cfd, maxfd, activity, i, valread;
    int client_socket[MAX_CLIENTS] = {0};
    fd_set readfds, rdtemp;
    struct sockaddr_in addr, cliaddr;
    socklen_t cliLen = sizeof(cliaddr);
    char buffer[1024];

    // 创建监听的套接字
    if ((lfd = socket(AF_INET, SOCK_STREAM, 0)) == 0) {
        perror("socket failed");
        exit(EXIT_FAILURE);
    }

    // 设置服务器地址结构
    addr.sin_family = AF_INET;
    addr.sin_port = htons(PORT);
    addr.sin_addr.s_addr = INADDR_ANY;

    // 绑定套接字到本地IP和端口
    if (bind(lfd, (struct sockaddr *)&addr, sizeof(addr)) < 0) {
        perror("bind failed");
        exit(EXIT_FAILURE);
    }

    // 开始监听连接
    if (listen(lfd, 128) < 0) {
        perror("listen");
        exit(EXIT_FAILURE);
    }

    printf("Waiting for connections on port %d...\n", PORT);

    // 初始化文件描述符集合
    FD_ZERO(&readfds);
    FD_SET(lfd, &readfds);  // 将监听套接字加入集合
    maxfd = lfd;            // 初始时最大的文件描述符是监听套接字

    while (1) {
        // 阻塞等待文件描述符就绪
        rdtemp = readfds;
        activity = select(maxfd + 1, &rdtemp, NULL, NULL, NULL);

        if (activity < 0 && errno != EINTR) {
            perror("select error");
        }

        // 处理新连接
        if (FD_ISSET(lfd, &rdtemp)) {
            if ((cfd = accept(lfd, (struct sockaddr *)&cliaddr, &cliLen)) < 0) {
                perror("accept failed");
                exit(EXIT_FAILURE);
            }

            printf("New connection: socket fd is %d, IP is : %s, port : %d\n",
                   cfd, inet_ntoa(cliaddr.sin_addr), ntohs(cliaddr.sin_port));

            // 将新的通信套接字加入集合
            for (i = 0; i < MAX_CLIENTS; i++) {
                if (client_socket[i] == 0) {
                    client_socket[i] = cfd;
                    break;
                }
            }

            // 更新最大文件描述符
            if (cfd > maxfd) {
                maxfd = cfd;
            }

            FD_SET(cfd, &readfds);  // 将新连接的套接字加入集合
        }

        // 处理客户端消息
        for (i = 0; i < MAX_CLIENTS; i++) {
            cfd = client_socket[i];

            if (FD_ISSET(cfd, &rdtemp)) {
                valread = read(cfd, buffer, sizeof(buffer));

                if (valread == 0) {
                    // 客户端关闭连接
                    getpeername(cfd, (struct sockaddr *)&cliaddr, &cliLen);
                    printf("Host disconnected, IP %s, port %d, fd %d\n",
                           inet_ntoa(cliaddr.sin_addr), ntohs(cliaddr.sin_port), cfd);

                    close(cfd);
                    FD_CLR(cfd, &readfds);  // 从集合中移除套接字
                    client_socket[i] = 0;
                } else {
                    // 回显客户端消息
                    buffer[valread] = '\0';
                    printf("Client %s, fd %d: %s\n", inet_ntoa(cliaddr.sin_addr), cfd, buffer);

                    // 发送回显数据给客户端
                    send(cfd, buffer, strlen(buffer), 0);
                }
            }
        }
    }

    return 0;
}

在上面的代码中,创建了两个fd_set变量,用于保存要检测的读集合:

// 初始化检测的读集合
fd_set rdset;        //要检测的位置集合
fd_set rdtemp;       //这些检测位置集合中有变化的的集合,就是有就绪状态的文件描述符

rdset用于保存要检测的原始数据,这个变量不能作为参数传递给select函数,因为在函数内部这个变量中的值会被内核修改,函数调用完毕返回之后,里边就不是原始数据了,大部分情况下是值为1的标志位变少了,不可能每一轮检测,所有的文件描述符都是就行的状态。因此需要通过rdtemp变量将原始数据传递给内核,select() 调用完毕之后再将内核数据传出,这两个变量的功能是不一样的。

客户端代码:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <arpa/inet.h>

int main()
{
    // 1. 创建用于通信的套接字
    int fd = socket(AF_INET, SOCK_STREAM, 0);
    if(fd == -1)
    {
        perror("socket");
        exit(0);
    }

    // 2. 连接服务器
    struct sockaddr_in addr;
    addr.sin_family = AF_INET;     // ipv4
    addr.sin_port = htons(9999);   // 服务器监听的端口, 字节序应该是网络字节序
    inet_pton(AF_INET, "127.0.0.1", &addr.sin_addr.s_addr);
    int ret = connect(fd, (struct sockaddr*)&addr, sizeof(addr));
    if(ret == -1)
    {
        perror("connect");
        exit(0);
    }

    // 通信
    while(1)
    {
        // 读数据
        char recvBuf[1024];
        // 写数据
        // sprintf(recvBuf, "data: %d\n", i++);
        fgets(recvBuf, sizeof(recvBuf), stdin);
        write(fd, recvBuf, strlen(recvBuf)+1);
        // 如果客户端没有发送数据, 默认阻塞
        read(fd, recvBuf, sizeof(recvBuf));
        printf("recv buf: %s\n", recvBuf);
        sleep(1);
    }

    // 释放资源
    close(fd); 

    return 0;
}

虽然使用select这种IO多路转接技术可以降低系统开销,提高程序效率,但是它也有局限性:

  • 待检测集合(第2、3、4个参数)需要频繁的在用户区和内核区之间进行数据的拷贝,效率低
  • 内核对于select传递进来的待检测集合的检测方式是线性的
  • 如果集合内待检测的文件描述符很多,检测效率会比较低
  • 如果集合内待检测的文件描述符相对较少,检测效率会比较高
  • 使用select能够检测的最大文件描述符个数有上限,默认是1024,这是在内核中被写死了的。

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

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

相关文章

【Java】中的List集合

目录 一、什么是List集合二、List的常用方法List的初始化元素操作1.添加元素2.删除元素3.修改元素4.查询元素 三、List集合的遍历1.for循环遍历2.增强for循环3.迭代器遍历 一、什么是List集合 List集合是最常用的一种数据结构之一。它具有动态扩容、元素添加、删除和查询等基础…

MySQL--索引(2)

InnoDB 1.索引类型 主键索引(Primary Key) 数据表的主键列使用的就是主键索引。 一张数据表有只能有一个主键&#xff0c;并且主键不能为 null&#xff0c;不能重复。 在 mysql 的 InnoDB 的表中&#xff0c;当没有显示的指定表的主键时&#xff0c;InnoDB 会自动先检查表中是…

IC秋招RTL代码合集

一 全加器和半加器 全加器 module full_adder1(input Ai, Bi, Ci,output So, Co);assign So Ai ^ Bi ^ Ci ;assign Co (Ai & Bi) | (Ci & (Ai | Bi)); endmodule module full_adder1(input Ai, Bi, Cioutput So, Co);assign {Co, So} Ai Bi Ci ; endm…

Wi-SUN无线通信技术 — 大规模分散式物联网应用首选

引言 在数字化浪潮的推动下&#xff0c;物联网&#xff08;IoT&#xff09;正逐渐渗透到我们生活的方方面面。Wi-SUN技术以其卓越的性能和广泛的应用前景&#xff0c;成为了大规模分散式物联网应用的首选。本文将深入探讨Wi-SUN技术的市场现状、核心优势、实际应用中的案例以及…

AndroidStudio 编辑xml布局文件卡死问题解决

之前项目编写的都是正常&#xff0c;升级AndroidStudio后编辑布局文件就卡死&#xff0c;还以为是AndroidStudio文件。 其实不然&#xff0c;我给整个项目增加了版权声明。所以全部跟新后&#xff0c;布局文件也增加了版权声明。估计AndroidStudio在 解析布局文件时候因为有版…

信号【Linux】

文章目录 信号处理方式&#xff08;信号递达&#xff09;前后台进程 终端按键产生信号kill系统调用接口向进程发信号阻塞信号sigset_tsigprocmasksigpending内核态与用户态&#xff1a;内核空间与用户空间内核如何实现信号的捕捉 1、信号就算没有产生&#xff0c;进程也必须识别…

Django—admin后台管理

Django官网 https://www.djangoproject.com/ 如果已经有了Django跳过这步 安装Django&#xff1a; 如果你还没有安装Django&#xff0c;可以通过Python的包管理器pip来安装&#xff1a; pip install django 创建项目&#xff1a; 使用Django创建一个新的项目&#xff1a; …

敲详细的springboot中使用RabbitMQ的源码解析

这里介绍的源码主要是涉及springboot框架下的rabbitmq客户端代码&#xff08;具体在springframework.amqp.rabbit包下&#xff0c;区分一下不由springboot直接接管的spring-rabbit的内容&#xff09;&#xff0c;springboot基于RabbitMQ的Java客户端建立了简便易用的框架。 sp…

jmeter实战(1)- Mac环境安装

一、安装 JDK 这个就不介绍了&#xff0c;本地自行安装 JDK 并且配置好环境变量 二、安装 Jmeter 1. 下载地址 —> 下载链接点击这里 2. 选择合适的版本下载 3. 解压到本地目录 解压后&#xff0c;会得到下面的目录文件&#xff1a; 输入cd bin&#xff0c;进入到bin…

OpenCV 直方图概念,直方图均衡化原理详解

文章目录 直方图相关概念颜色灰度级作用应用场景 C 使用OpenCV绘制直方图单通道直方图关键代码分析&#xff1a;calcHist函数分析使用OpenCV API来绘制直方图 效果图&#xff1a; 彩色三通道直方图效果图&#xff1a; 直方图均衡化概念均衡化作用均衡化效果均衡化数学原理步骤数…

项目实战二 HIS项目

目标&#xff1a; 项目的操作流程&#xff1a; 开发体系 前端开发&#xff1a;负责页面的编写 HTML CSS JavaScript 后端开发&#xff1a;看不到 摸不着的功能 常用开发语言 PHP JAVA Python 框架 &#xff1a; 半成品 做好的功能模块 版本控制 Git 分布式版本控…

vxe-table——实现切换页码时排序状态的回显问题(ant-design+elementUi中table排序不同时回显的bug)——js技能提升

之前写的后台管理系统&#xff0c;都是用的antdelement&#xff0c;table组件中的【排序】问题是有一定的缺陷的。 想要实现的效果&#xff1a; antv——table组件一次只支持一个参数的排序 如下图&#xff1a; 就算是可以自行将排序字段拼接到列表接口的入参中&#xff0c…

【中项】系统集成项目管理工程师-第4章 信息系统架构-4.3应用架构

前言&#xff1a;系统集成项目管理工程师专业&#xff0c;现分享一些教材知识点。觉得文章还不错的喜欢点赞收藏的同时帮忙点点关注。 软考同样是国家人社部和工信部组织的国家级考试&#xff0c;全称为“全国计算机与软件专业技术资格&#xff08;水平&#xff09;考试”&…

vue3 -layui项目-左侧导航菜单栏

1.创建目录结构 进入cmd,先cd到项目目录&#xff08;项目vue3-project&#xff09; cd vue3-project mkdir -p src\\views\\home\\components\\menubar 2.创建组件文件 3.编辑menu-item-content.vue <template><template v-if"item.icon"><lay-ic…

计算机网络八股文(后续更新)

文章目录 一、计算机网络体系结构1、计算机网络的各层协议及作用&#xff1f; 二、物理层三、数据链路层四、网络层五、传输层1、TCP和UDP的区别&#xff1f;2、UDP 和 TCP 对应的应用场景是什么&#xff1f;3、详细介绍一下 TCP 的三次握手机制4、为什么需要三次握手&#xff…

Electron 结合 Selenium + chromedriver 驱动服务实现浏览器多开

背景 在调研浏览器多开的过程中&#xff0c;electron 有自带的 browserview&#xff0c;webview&#xff0c;但是上面两个受制于 electron 内核版本限制&#xff0c;升级不够灵活&#xff0c;对新版的网页支持可能不及时&#xff0c;甚至不兼容&#xff0c;必须通过发布新的客…

MSP430M03507最小系统板的keil环境搭配,用keil编辑ti单片机

转载自嘉立创MSP430M03507开发手册 这篇文章只是因为我的keil版本与嘉立创的不一样&#xff0c;所以添加了我自己遇到的问题解析 先说说为什么要用keil编辑&#xff0c;因为ti单片机自己的ccs编译环境需要对应仿真器&#xff0c;那个加芯片都240了&#xff0c;哪有那么多钱买…

JAVA开发工具IDEA如何连接操作数据库

一、下载驱动 下载地址&#xff1a;【免费】mysql-connector-j-8.2.0.jar资源-CSDN文库 二、导入驱动 鼠标右击下载到IDEA中的jar包&#xff0c;选择Add as Library选项 如图就导入成功 三、加载驱动 Class.forName("com.mysql.cj.jdbc.Driver"); 四、驱动管理…

Bootstrap5 Navbar多级下拉框

实现目标&#xff1a; 1、访问 Bootstrap5-navbar 2、修改dropdown为多级 <!DOCTYPE HTML> <html lang"en-US"> <head><meta charset"UTF-8"><title></title><link rel"stylesheet" href"https…

视频汇聚平台EasyCVR启动出现报错“cannot open shared object file”的原因排查与解决

安防视频监控EasyCVR安防监控视频系统采用先进的网络传输技术&#xff0c;支持高清视频的接入和传输&#xff0c;能够满足大规模、高并发的远程监控需求。EasyCVR平台支持多种视频流的外部分发&#xff0c;如RTMP、RTSP、HTTP-FLV、WebSocket-FLV、HLS、WebRTC、fmp4等&#xf…