基于 Apache Commons Pool 实现的 gRPC 连接池管理类 GrpcChannelPool 性能分析与优化

基于 Apache Commons Pool 实现的 gRPC 连接池管理类 GrpcChannelPool 性能分析与优化

1. 输出关键信息的代码示例

日志记录方法

使用以下代码记录连接池的关键信息,帮助分析连接池的状态和性能瓶颈:

import org.apache.commons.pool2.impl.GenericObjectPool;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class GrpcChannelPoolAnalyzer {
    private static final Logger logger = LoggerFactory.getLogger(GrpcChannelPoolAnalyzer.class);
    private final GenericObjectPool<GrpcChannel> grpcChannelPool;

    public GrpcChannelPoolAnalyzer(GenericObjectPool<GrpcChannel> grpcChannelPool) {
        this.grpcChannelPool = grpcChannelPool;
    }

    public void logPoolStatus() {
        logger.info("=== GrpcChannelPool Status ===");
        logger.info("Num Active: {}", grpcChannelPool.getNumActive());
        logger.info("Num Idle: {}", grpcChannelPool.getNumIdle());
        logger.info("Num Waiters: {}", grpcChannelPool.getNumWaiters());
        logger.info("Total Borrowed Count: {}", grpcChannelPool.getBorrowedCount());
        logger.info("Total Created Count: {}", grpcChannelPool.getCreatedCount());
        logger.info("Total Returned Count: {}", grpcChannelPool.getReturnedCount());
        logger.info("Total Destroyed Count: {}", grpcChannelPool.getDestroyedCount());
    }
}

调用日志记录

public static void main(String[] args) {
    // 创建连接池
    GenericObjectPool<GrpcChannel> grpcChannelPool = new GenericObjectPool<>(new GrpcChannelFactory());

    // 初始化分析器
    GrpcChannelPoolAnalyzer analyzer = new GrpcChannelPoolAnalyzer(grpcChannelPool);

    // 定期输出日志
    analyzer.logPoolStatus();
}

2. 关键方法的含义和影响分析

A. getBorrowedCount()

  • 含义
    • 自池创建以来,成功借出的对象总数。
    • 表示连接池处理了多少次借用请求。
  • 可能的影响
    • 如果 getBorrowedCount 很高,说明连接池被频繁借用。这可能导致以下问题:
      1. 如果连接池配置的 maxTotal 过低,导致线程排队等待可用连接。
      2. 如果池中连接被频繁销毁或验证失败,导致额外的连接创建开销。
    • 借用连接时慢可能是由于等待连接归还或新建连接的时间过长。

B. getCreatedCount()

  • 含义
    • 自池创建以来,总共创建的对象数量。
    • 包括了所有当前活跃、空闲和已销毁的对象。
  • 可能的影响
    • 如果 getCreatedCount 很高,表明连接池频繁创建新连接,可能是因为:
      1. 空闲连接销毁过快:池中配置了较小的 maxIdle 或空闲对象检测频率较高,导致连接被频繁销毁。
      2. 验证失败:连接在借用时被标记为无效,从而需要创建新连接。
      3. 连接使用时间长:池内的连接未及时归还,导致新连接不断创建。
    • 借用连接时慢可能是由于新连接创建速度较慢(如涉及网络请求、SSL 加密等复杂逻辑)。

C. getNumActive()

  • 含义
    • 当前正在被使用的连接数。
  • 可能的影响
    • 如果 getNumActive 接近或等于 maxTotal
      1. 新的借用请求将被阻塞,等待有连接归还或新建。
      2. 借用延迟的时间取决于前一个连接释放的速度或新连接创建时间。

D. getNumIdle()

  • 含义
    • 当前池中空闲的连接数。
  • 可能的影响
    • 如果 getNumIdle 为 0,意味着没有可立即借用的连接。
    • 此时,借用请求会被阻塞,直到:
      1. 一个连接被归还。
      2. 一个新连接被创建。
    • 借用连接时慢通常与空闲连接耗尽直接相关。

E. getNumWaiters()

  • 含义
    • 当前等待借用连接的线程数。
  • 可能的影响
    • 如果等待线程数较多,表明连接池无法及时满足请求。
    • 这种情况可能因为 maxTotal 设置过低或连接释放速度过慢导致。

F. getDestroyedCount()

  • 含义
    • 自池创建以来,被销毁的连接总数。
  • 可能的影响
    • 如果销毁的连接数较高,可能导致借用连接时需要频繁创建新连接,从而增加延迟。
    • 检查空闲对象的检测配置是否合理(如 timeBetweenEvictionRunsMillisminEvictableIdleTimeMillis)。

3. 借用对象慢的分析原因

根据以上信息,以下是常见可能的原因及解决方案:

原因 1:空闲连接不足

  • 表现getNumIdle 为 0。
  • 分析:池中没有空闲连接供借用,新的借用请求需要等待归还或创建新连接。
  • 解决方案
    1. 增加 minIdlemaxIdle,确保有足够的空闲连接。
    2. 减少连接创建的开销(优化 makeObject() 方法)。

原因 2:线程等待超时

  • 表现getNumWaiters 高,getNumActive 接近或等于 maxTotal
  • 分析:借用请求太多,超过了池的最大容量。
  • 解决方案
    1. 增加 maxTotal,允许池处理更多并发连接。
    2. 优化业务逻辑,减少连接使用时间。

原因 3:连接频繁销毁和创建

  • 表现getCreatedCountgetDestroyedCount 较高。
  • 分析:可能因连接超时、验证失败或空闲销毁策略不合理导致。
  • 解决方案
    1. 调整连接池的空闲销毁参数(如 timeBetweenEvictionRunsMillisminEvictableIdleTimeMillis)。
    2. 优化连接验证逻辑,减少验证失败的情况。

原因 4:连接创建耗时

  • 表现getNumActive 达到 maxTotalgetNumIdle 为 0。
  • 分析:新连接的创建时间太长,可能因网络延迟、认证复杂或初始化慢。
  • 解决方案
    1. 优化连接的创建过程(减少网络调用、使用连接池化资源)。
    2. 预热池:配置 minIdle 和定期检测任务。

4. 日志输出与示例分析

假设运行时日志输出如下:

INFO - === GrpcChannelPool Status ===
INFO - Num Active: 10
INFO - Num Idle: 0
INFO - Num Waiters: 95
INFO - Total Borrowed Count: 500
INFO - Total Created Count: 520
INFO - Total Destroyed Count: 10

分析

  1. Num Idle = 0

    • 空闲连接不足,导致新的借用请求需要等待。
  2. Num Waiters = 95

    • 表示 95 个线程正在等待借用连接,系统压力较大。
  3. Total Created Count = 520,Total Destroyed Count = 10

    • 表明创建频率高,销毁频率低,可能是空闲检测频率较低或借用时触发新连接的创建逻辑。

解决方案

  • 增加 maxTotal(如 50)。
  • 设置合理的 minIdle(如 10)并启用预热机制。
  • 优化连接创建速度,减少延迟。

5. 如何获取 maxTotal 与其关系

获取 maxTotal 的方法

maxTotal 是连接池中可同时活跃对象的最大数量。在 Apache Commons Pool 中,通过 GenericObjectPoolConfig 设置并管理此参数。

获取 maxTotal 的代码

如果你有一个 GenericObjectPool 实例,可以通过以下代码获取 maxTotal 的值:

int maxTotal = grpcChannelPool.getMaxTotal();
System.out.println("Max Total: " + maxTotal);

maxTotalNumActive 的关系

定义
  • maxTotal:

    • 定义了连接池中允许同时活跃的最大对象数量。
    • 当活跃对象数量达到 maxTotal 时,新的借用请求会被阻塞,直到有对象归还或超时。
  • NumActive:

    • 表示当前池中被借用(活跃使用)的对象数量。
    • NumActive 的值始终小于或等于 maxTotal
关系描述
  1. NumActive ≤ maxTotal

    • NumActive 是当前实际使用的对象数量,受 maxTotal 的限制。
    • 如果 NumActive 达到 maxTotal,连接池不会再创建新对象,而是让新的请求进入等待状态。
  2. 连接池满的场景

    • NumActive == maxTotalNumIdle == 0 时,连接池满载,新的借用请求将进入等待队列,直到有对象被归还或超时。
  3. 调整 maxTotal 的影响

    • 增加 maxTotal:允许池支持更多并发请求,但需要足够的系统资源(如数据库连接数)。
    • 减少 maxTotal:限制池的最大并发能力,可能导致更多请求进入等待队列。

示例分析

代码示例

以下代码展示如何获取 maxTotal 和检查其与 NumActive 的关系:

import org.apache.commons.pool2.impl.GenericObjectPool;

public class PoolStatusChecker {
    public static void logPoolStatus(GenericObjectPool<?> pool) {
        int maxTotal = pool.getMaxTotal();
        int numActive = pool.getNumActive();
        int numIdle = pool.getNumIdle();

        System.out.println("Max Total: " + maxTotal);
        System.out.println("Num Active: " + numActive);
        System.out.println("Num Idle: " + numIdle);

        if (numActive == maxTotal && numIdle == 0) {
            System.out.println("Connection pool is at full capacity.");
        }
    }
}
运行结果示例

假设运行时的池状态如下:

Max Total: 10
Num Active: 10
Num Idle: 0

分析

  • 当前活跃连接数 NumActive = 10 已经达到 maxTotal
  • 此时,如果有新的借用请求,将进入等待队列,可能导致延迟。

总结

  • maxTotal 定义了池中最大同时活跃对象的数量,而 NumActive 是当前实际使用的对象数量。
  • NumActive 始终小于或等于 maxTotal,当 NumActive == maxTotalNumIdle == 0 时,新的借用请求将进入等待状态。
  • 根据业务需求动态调整 maxTotal,并结合 NumActiveNumWaiters 的监控数据,优化连接池配置。

GrpcChannelPool 性能分析与数学关系

1. 关键值的意义和关系

活跃连接数 (NumActive)

  • 当前正在使用的连接数量。
  • NumActive ≤ maxTotal(活跃连接不能超过最大连接数)。

空闲连接数 (NumIdle)

  • 当前池中未被使用的连接数量。
  • NumIdle + NumActive ≤ maxTotal(空闲和活跃连接之和不能超过池中允许的最大连接数)。

等待线程数 (NumWaiters)

  • 当前等待借用连接的线程数。
  • NumActive == maxTotalNumIdle == 0 时,新的请求进入等待队列,此时 NumWaiters > 0

最大连接数 (maxTotal)

  • 池中允许的最大同时活跃连接数。
  • 限制了 NumActiveNumIdle 的上限。

2. 累积统计值

借用总数 (Total Borrowed Count)

  • 表示自池创建以来成功借出的对象总数。
  • Total Borrowed Count ≥ NumActive(历史借用次数必然大于或等于当前活跃数)。

创建总数 (Total Created Count)

  • 表示自池创建以来创建的对象总数。
  • Total Created Count ≥ NumActive + NumIdle(当前池中活跃和空闲连接数量必须由创建的连接数提供)。

归还总数 (Total Returned Count)

  • 表示自池创建以来归还的对象总数。
  • Total Returned Count = Total Borrowed Count - NumActive(归还的连接数等于借用总数减去当前未归还的活跃连接数)。

销毁总数 (Total Destroyed Count)

  • 表示自池创建以来销毁的对象总数。
  • Total Destroyed Count = Total Created Count - (NumActive + NumIdle)(销毁的连接数等于创建的连接数减去池中当前剩余的活跃和空闲连接)。

3. 主要数学运算关系

在这里插入图片描述


4. 示例分析

假设运行时池的状态如下:

  • maxTotal = 10
  • NumActive = 6
  • NumIdle = 2
  • NumWaiters = 3
  • Total Borrowed Count = 50
  • Total Created Count = 15
  • Total Returned Count = 44
  • Total Destroyed Count = 7

验证数学关系:

在这里插入图片描述


5. 如何利用这些关系分析性能问题

通过以上数学关系,可以监控和诊断连接池的问题。例如:

池容量不足

  • 表现
    • 如果 NumActive + NumIdle 长期接近或等于 maxTotal,且 NumWaiters > 0
  • 原因
    • 连接池容量不足,导致大量线程排队。
  • 解决方案
    • 增加 maxTotal

连接泄漏

  • 表现
    • 如果 Total Borrowed Count - Total Returned Count ≠ NumActive
  • 原因
    • 存在未归还的连接。
  • 解决方案
    • 检查业务代码,确保每次借用的连接都正确归还。

连接频繁创建销毁

  • 表现
    • 如果 Total Destroyed Count 很高。
  • 原因
    • 连接被频繁销毁,可能是因为空闲策略不合理。
  • 解决方案
    • 调整 minIdle 和空闲销毁参数。

6. 总结

这些值之间的数学关系提供了诊断连接池运行状态的重要依据。通过监控和分析这些关系,可以:

  • 优化连接池的配置(如 maxTotalminIdle 等)。
  • 发现性能瓶颈(如等待时间过长、连接不足等)。
  • 及时修复问题(如连接泄漏或资源浪费)。

GrpcChannelPool 预热机制详解

1. 预热机制的概念

预热机制是指在连接池启动时或者空闲连接不足时,预先创建一定数量的连接(由 minIdle 参数指定),以确保在请求到来时能够快速响应,避免因为连接创建而导致延迟。

在 Apache Commons Pool 中,预热机制通过以下参数实现:

  1. minIdle: 定义了连接池中保持的最小空闲连接数。如果空闲连接少于这个值,连接池会主动补充连接。
  2. timeBetweenEvictionRunsMillis: 定义了空闲对象检测线程的运行周期。如果设置了该值,后台线程会定期运行,确保空闲连接数不低于 minIdle
  3. numTestsPerEvictionRun: 定义每次检测时要检查的对象数量。

2. 如何启用预热机制

步骤 1: 设置 minIdle

  • 定义连接池的最小空闲连接数。例如,如果你的系统需要至少保持 5 个连接可以快速响应:
    poolConfig.setMinIdle(5);
    

步骤 2: 启用空闲检测线程

  • 设置 timeBetweenEvictionRunsMillis,让连接池定期检查和补充空闲连接。例如:
    poolConfig.setTimeBetweenEvictionRunsMillis(10000); // 每 10 秒运行一次检测任务
    

步骤 3: 确保足够的空闲连接

  • 设置 numTestsPerEvictionRun,控制每次检测的对象数量。通常设置为一个较大的值(如 -1,表示检测所有对象):
    poolConfig.setNumTestsPerEvictionRun(-1);
    

可选参数

  • testWhileIdle:
    • 如果设置为 true,连接池会在补充空闲连接时验证连接是否有效。
    • 适合场景:如果连接容易失效(例如网络中断),启用此参数可提高连接池的可靠性。

3. 完整的预热机制代码示例

以下是一个示例,展示如何为连接池启用预热机制:

import org.apache.commons.pool2.impl.GenericObjectPoolConfig;

public class GrpcChannelPoolConfig {
    public static GenericObjectPoolConfig<Object> createConfig() {
        GenericObjectPoolConfig<Object> poolConfig = new GenericObjectPoolConfig<>();
        
        // 设置连接池大小
        poolConfig.setMaxTotal(50);    // 最大连接数
        poolConfig.setMinIdle(5);      // 最小空闲连接数
        poolConfig.setMaxIdle(10);     // 最大空闲连接数
        
        // 启用预热机制
        poolConfig.setTimeBetweenEvictionRunsMillis(10000); // 每 10 秒检测一次
        poolConfig.setNumTestsPerEvictionRun(-1);           // 每次检测所有对象
        poolConfig.setTestWhileIdle(true);                  // 检查空闲连接有效性
        
        return poolConfig;
    }
}

4. 预热机制的优点

  1. 减少首次延迟:在请求到达前已创建好足够的连接,无需等待连接创建。
  2. 提高系统响应能力:尤其在负载波动较大的场景下,预热机制能缓解连接不足的问题。
  3. 提升可靠性:通过定期检测和补充空闲连接,确保池中连接始终处于健康状态。

5. 注意事项

  1. 监控连接数量

    • 如果设置了过大的 minIdle,可能导致资源浪费(如内存、数据库连接等)。
    • 建议通过监控工具(如 JMX)观察实际连接使用情况,动态调整参数。
  2. 创建时间消耗

    • 如果连接创建耗时较长(如 SSL 或远程服务连接),建议适当增加 timeBetweenEvictionRunsMillis 的间隔,避免频繁创建。
  3. 避免过度销毁

    • 配置 maxIdleminIdle 的差距不宜过大,防止频繁的连接创建和销毁。

6. 总结

预热机制的核心是通过配置 minIdle 和空闲检测线程定期补充连接,以确保空闲连接数不低于 minIdle。启用预热机制的关键配置是:

  1. setMinIdle:定义最小空闲连接数。
  2. setTimeBetweenEvictionRunsMillis:定期检测周期。
  3. 可选:启用 testWhileIdle 提高连接有效性检测。

这些参数协同工作,可以显著提高连接池的性能和响应能力。



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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

矩阵碰一碰发视频的视频剪辑功能源码搭建,支持OEM

在短视频创作与传播领域&#xff0c;矩阵碰一碰发视频结合视频剪辑功能&#xff0c;为用户带来了高效且富有创意的内容产出方式。这一功能允许用户通过碰一碰 NFC 设备触发视频分享&#xff0c;并在分享前对视频进行个性化剪辑。以下将详细阐述该功能的源码搭建过程。 一、技术…

CClinkIEfield Basic转Modbus TCP网关模块连接三菱FX5U PLC

捷米特JM-CCLKIE-TCP是自主研发的一款CCLINK IE FB从站功能的通讯网关。该产品主要功能是将各种 MODBUS-TCP 设备接入到 CCLINK IE FB网络中。 捷米特JM-CCLKIE-TCP网关连接到CCLINK IE FB总线中做为从站使用&#xff0c;连接到 MODBUS-TCP 总线中做为主站或从站使用。 为了打破…

农产品智慧物流系统

本文结尾处获取源码。 本文结尾处获取源码。 本文结尾处获取源码。 一、相关技术 后端&#xff1a;Java、JavaWeb / Springboot。前端&#xff1a;Vue、HTML / CSS / Javascript 等。数据库&#xff1a;MySQL 二、相关软件&#xff08;列出的软件其一均可运行&#xff09; I…

设计模式-结构型-桥接模式

1. 什么是桥接模式&#xff1f; 桥接模式&#xff08;Bridge Pattern&#xff09; 是一种结构型设计模式&#xff0c;它旨在将抽象部分与实现部分分离&#xff0c;使它们可以独立变化。通过这种方式&#xff0c;系统可以在抽象和实现两方面进行扩展&#xff0c;而无需相互影响…

后台管理系统引导功能的实现

引导是软件中经常见到的一个功能&#xff0c;无论是在后台项目还是前台或者是移动端项目中。 那么对于引导页而言&#xff0c;它是如何实现的呢&#xff1f;通常情况下引导页是通过 聚焦 的方式&#xff0c;高亮一块视图&#xff0c;然后通过文字解释的形式来告知用户该功能的作…

现场展示deepseek VS openAI o1模型大对比

DeepSeek-V3 模型的发布在 AI 领域引起了广泛关注。作为一款拥有 6850 亿参数的混合专家&#xff08;MoE&#xff09;语言模型&#xff0c;DeepSeek-V3 在多个基准测试中表现出色&#xff0c;甚至超越了一些闭源模型。其在 Aider 代码能力排行榜上的正确率达到 48.4%&#xff0…

Golang的并发编程框架比较

# Golang的并发编程框架比较 中的并发编程 在现代软件开发中&#xff0c;处理高并发的能力愈发重要。Golang作为一门支持并发编程的编程语言&#xff0c;提供了丰富的并发编程框架和工具&#xff0c;使得开发者能够更轻松地处理并发任务。本文将介绍Golang中几种常用的并发编程…

SSL,TLS协议分析

写在前面 工作中总是会接触到https协议&#xff0c;也知道其使用了ssl&#xff0c;tls协议。但对其细节并不是十分的清楚。所以&#xff0c;就希望通过这篇文章让自己和读者朋友们都能对这方面知识有更清晰的理解。 1&#xff1a;tls/ssl协议的工作原理 1.1&#xff1a;设计的…

网络安全-XSS跨站脚本攻击(基础篇)

漏洞扫描的原理 1.跨站脚本攻击介绍 xss跨站脚本攻击&#xff1a; xSS 全称&#xff08;Cross site Scripting &#xff09;跨站脚本攻击&#xff0c;是最常见的Web应用程序安全漏洞之一&#xff0c;位于OWASP top 10 2013/2017年度分别为第三名和第七名&#xff0c;XSS是指攻…

SpringBoot之核心配置

学习目标&#xff1a; 1.熟悉Spring Boot全局配置文件的使用 2.掌握Spring Boot配置文件属性值注入 3.熟悉Spring Boot自定义配置 4.掌握Profile多环境配置 5.了解随机值设置以及参数间引用 1.全局配置文件 Spring Boot使用 application.properties 或者application.yaml 的文…

【Word_笔记】Word的修订模式内容改为颜色标记

需求如下&#xff1a;请把修改后的部分直接在原文标出来&#xff0c;不要采用修订模式 步骤1&#xff1a;打开需要转换的word后&#xff0c;同时按住alt和F11 进入&#xff08;Microsoft Visual Basic for Appliations&#xff09; 步骤2&#xff1a;插入 ---- 模块 步骤3&…

【C++】字符数|组与字符串的深度解析

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: C 文章目录 &#x1f4af;前言&#x1f4af;一、字符数组的基本概念1. 什么是字符数组&#xff1f;2. C语言风格字符串的特点 &#x1f4af;二、字符数组的初始化1. 字符串直接赋值2. 按字符逐个赋值数据对比示例 &am…

计算机网络——网络层—IP数据报与分片

一、IP 数据报的格式 • 一个 IP 数据报由首部和数据两部分组成。 • 首部的前一部分是固定长度&#xff0c;共 20 字节&#xff0c;是所有 IP 数据报必须具有的。 • 在首部的固定部分的后面是一些可选字段&#xff0c;其长度是可变的。 IP 数据报首部的固定部分中的各字段 版…

【Python学习(八)——异常处理】

Python学习&#xff08;八&#xff09;——异常处理 本文介绍了异常处理的知识&#xff0c;仅作为本人学习时记录&#xff0c;感兴趣的初学者可以一起看看&#xff0c;欢迎评论区讨论&#xff0c;一起加油鸭~~~ 心中默念&#xff1a;Python 简单好学&#xff01;&#xff01;&…

Python 爬虫验证码识别

在我们进行爬虫的过程中&#xff0c;经常会碰到有些网站会时不时弹出来验证码识别。我们该如何解决呢&#xff1f;这里分享 2 种我尝试过的方法。 0.验证码示例 1.OpenCV pytesseract 使用 Python 中的 OpenCV 库进行图像预处理&#xff08;边缘保留滤波、灰度化、二值化、…

[离线数仓] 总结二、Hive数仓分层开发

接 [离线数仓] 总结一、数据采集 5.8 数仓开发之ODS层 ODS层的设计要点如下: (1)ODS层的表结构设计依托于从业务系统同步过来的数据结构。 (2)ODS层要保存全部历史数据,故其压缩格式应选择压缩比率,较高的,此处选择gzip。 CompressedStorage - Apache Hive - Apac…

GraphQL:强大的API查询语言

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

vue js实现时钟以及刻度效果

2025.01.08今天我学习如何用js实现时钟样式&#xff0c;效果如下&#xff1a; 一、html代码如下&#xff1a; <template><!--圆圈--><div class"notice_border"><div class"notice_position notice_name_class" v-for"item in …

Docker入门之docker基本命令

Docker入门之docker基本命令 官方网站&#xff1a;https://www.docker.com/ 1. 拉取官方镜像并创建容器&#xff08;以redis为例&#xff09; 拉取官方镜像 docker pull redis# 如果不需要添加到自定义网络使用这个命令&#xff0c;如需要&#xff0c;直接看第二步 docker r…

“深入浅出”系列之FFmpeg:(1)音视频开发基础

我的音视频开发大部分内容是跟着雷霄骅大佬学习的&#xff0c;所以笔记也是跟雷老师的博客写的。 一、音视频相关的基础知识 首先播放一个视频文件的流程如下所示&#xff1a; FFmpeg的作用就是将H.264格式的数据转换成YUV格式的数据&#xff0c;然后SDL将YUV显示到电脑屏幕上…