H.265 (HEVC) 网页播放:WebAssembly + FFmpeg 实现浏览器端的硬解/软解兼容方案

H.265 (HEVC) 网页播放:WebAssembly + FFmpeg 实现浏览器端的硬解/软解兼容方案

标签: #WebAssembly #FFmpeg #H.265 #WebCodecs #音视频开发 #前端性能


📉 前言:浏览器对 H.265 的“爱恨情仇”

为什么 <video src="video.h265.mp4"> 在 Chrome 里放不出来?
因为 H.265 的专利池太深了。只有 Safari (即使是 iOS) 和 Edge (需硬件支持) 原生支持较好。

我们的目标是构建一套混合解码方案

  1. 优先硬解 (WebCodecs):如果浏览器支持硬件加速(如 Chrome 94+ 的 WebCodecs),直接调用 GPU,性能起飞。
  2. 降级软解 (Wasm + FFmpeg):如果不支持,自动切换到 WebAssembly 版的 FFmpeg 进行 CPU 软解,利用 SIMD 指令集加速。

播放器架构图 (Mermaid):

🐢 方案 B: 软件解码

🚀 方案 A: 硬件解码

Yes

GPU 解码

No

Wasm指令

CPU 解码

视频流 (H.265/HEVC)

解封装 (Demuxer)

Encoded Packets

浏览器支持 WebCodecs?

WebCodecs API (VideoDecoder)

VideoFrame 对象

Web Worker

FFmpeg (Wasm + SIMD)

YUV420 数据

Canvas (WebGL)


🛠️ 一、 编译 FFmpeg 为 WebAssembly

这是最困难的一步。我们需要使用 Emscripten 将 C 语言编写的 FFmpeg 编译成 .wasm 文件。

关键编译参数:
为了性能,必须开启 Multithreading (多线程)SIMD (单指令多数据流)

# Docker 环境下编译示例 emcc \ -Llibavcodec -Llibavutil -Llibswscale \ -I. \ -o ffmpeg-decoder.js \ src/decoder.c \ -s WASM=1\ -s USE_PTHREADS=1\# 开启多线程 -s PTHREAD_POOL_SIZE=4\# 预分配线程池 -s SIMD=1\# 开启 SIMD 加速 (关键!) -s ALLOW_MEMORY_GROWTH=1\ -O3 # 最高优化等级

注意:src/decoder.c 是你需要编写的 C 语言胶水代码,用于暴露 FFmpeg 的 avcodec_send_packetavcodec_receive_frame 接口给 JS 调用。


🧬 二、 核心实现:Web Worker 中的解码循环

解码是 CPU 密集型任务,绝对不能放在主线程,否则页面会卡死。我们需要在 Web Worker 中运行 Wasm。

1. 初始化解码器 (Worker.js)
importScripts('ffmpeg-decoder.js');let decoderModule;let codecContext;// 初始化 Wasm 模块Module().then(module=>{ decoderModule = module;// 调用 C 导出的初始化函数 codecContext = decoderModule._init_h265_decoder();postMessage({type:'ready'});}); self.onmessage=function(e){const{ type, data }= e.data;if(type ==='decode'){// data 是包含 H.265 NALU 的 Uint8Array// 1. 将数据写入 Wasm 内存 heapconst ptr = decoderModule._malloc(data.length); decoderModule.HEAPU8.set(data, ptr);// 2. 调用解码// decode_frame 是 C 层封装的函数const ret = decoderModule._decode_frame(codecContext, ptr, data.length);// 3. 获取 YUV 数据并传回主线程if(ret ===0){// 从 Wasm 内存拷贝 Y, U, V 数据// 注意:使用 Transferable Objects (零拷贝) 提升性能const yuvData =getYUVFromWasm();postMessage({type:'render',frame: yuvData },[yuvData.buffer]);} decoderModule._free(ptr);}};

🎨 三、 高性能渲染:WebGL 处理 YUV

FFmpeg 解码出来的数据通常是 YUV420p 格式。
不要在 CPU 里把 YUV 转 RGB(这非常慢),要用 WebGL Shader 在 GPU 里转!

渲染流程:

  1. 创建 3 个 WebGL 纹理 (Texture),分别存放 Y、U、V 数据。
  2. 编写 Fragment Shader 进行矩阵转换。

Fragment Shader (GLSL):

precision mediump float; uniform sampler2D textureY; uniform sampler2D textureU; uniform sampler2D textureV; varying vec2 vTexCoord; void main() { float y = texture2D(textureY, vTexCoord).r; float u = texture2D(textureU, vTexCoord).r - 0.5; float v = texture2D(textureV, vTexCoord).r - 0.5; // YUV 转 RGB 公式 (BT.601) float r = y + 1.402 * v; float g = y - 0.34414 * u - 0.71414 * v; float b = y + 1.772 * u; gl_FragColor = vec4(r, g, b, 1.0); } 

⏱️ 四、 难点攻克:音画同步 (AV Sync)

视频能播了,但声音和画面对不上怎么办?
通常以 音频时钟 (Audio Clock) 为基准。

同步逻辑图 (Mermaid):

PTS < AudioTime (视频慢了)

PTS > AudioTime (视频快了)

PTS ≈ AudioTime (刚好)

渲染循环 Loop

当前视频帧 PTS vs 音频时间

丢帧 Skip Frame

等待 Delay

渲染到 Canvas

在 JS 主线程中:

functionrenderLoop(){const audioTime = audioContext.currentTime;const frame = frameBuffer[0];// 获取队列头部的帧if(!frame)returnrequestAnimationFrame(renderLoop);const diff = frame.pts - audioTime;if(diff <-0.03){// 视频落后超过 30ms -> 丢帧追赶 frameBuffer.shift();renderLoop();}elseif(diff >0.03){// 视频超前 -> 等待下一帧绘制requestAnimationFrame(renderLoop);}else{// 同步 -> 渲染drawYUV(frame); frameBuffer.shift();requestAnimationFrame(renderLoop);}}

📊 五、 性能优化清单

为了达到 1080p 甚至 4K 的流畅播放,以下优化必不可少:

  1. 开启 SIMD:在支持 SIMD 的浏览器上,软解性能提升 2-3 倍
  2. SharedArrayBuffer:在主线程和 Worker 之间共享内存,避免数据拷贝开销(需要配置 HTTP Header: Cross-Origin-Opener-Policy: same-origin)。
  3. OffscreenCanvas:将 Canvas 的控制权转移给 Worker,让渲染也在 Worker 线程完成,彻底解放主线程 UI。
  4. WebCodecs 优先:始终检测 VideoDecoder API。如果支持硬件解码,直接 bypass 掉 Wasm 模块,这是性能的降维打击。

🎯 总结

通过 Wasm + FFmpeg + WebGL,我们填补了浏览器 H.265 支持的空白。虽然软解 4K 依然吃力(主要受限于单线程 JS 调度和 CPU 算力),但在 720p/1080p 监控流、会议流场景下,这是一套成熟且工业级的解决方案。

Next Step:
现在的方案是基于现成 MP4 文件的。尝试结合 WebSocketWebRTC,接收实时的 H.265 NALU 流(如 RTSP 转 WS),实现一个低延迟的网页版安防监控播放器。

Read more

黑马点评中商铺地址查询前端显示问题

黑马点评中商铺地址查询前端显示问题

这个页面存在以下前端问题,这个页面中选择美食之后往下翻是不能自动滚动分页查询所有的美食店铺数据的。前端往下翻没反应 但是还有个很奇怪的点就是前端点击距离之后滚动查询第一页查询了两次,如下图,点击距离之后除了第一页的数据查询了两次之外就其他就正常了   最后看了好久才知道是查询页面数据大小的问题 我的后端代码如下: controller层 /** * 根据商铺类型分页查询商铺信息 * @param typeId 商铺类型 * @param current 页码 * @param x 经度 * @param y 纬度 * @return 商铺列表 */ @GetMapping("/of/type") public Result queryShopByType( @RequestParam("typeId") Integer typeId, @RequestParam(value = "current", defaultValue = "1") Integer current, @RequestParam(

当 AI 开始「打工仔」模式:OpenClaw 指挥多个 Agent

当 AI 开始「打工仔」模式:OpenClaw 指挥多个 Agent

当 AI 开始「打工仔」模式:OpenClaw 指挥多个 Agent 你有没有想过:让一个 AI 帮你算数学题,再让另一个 AI 把结果翻译成英文? 这听起来有点「多此一举」——毕竟一个 AI 就能同时做这两件事。但有时候,把任务拆分开来让不同的独立的 Agent 处理,是后续处理复杂任务的必要条件。 今天就分享一次有趣的实验:用OpenClaw 和 两个 Agent 串联完成一个完整的工作流。 前提条件 * openclaw: 2026.2.3 * 如果标记 😬,即用自然语言输入,在 webchat 中输入 * 如果标记 💻,即用命令行输入 如果标记 🔧,即背后的命令,不用管 💡 提示:用户只需用自然语言描述需求,无需手动执行底层命令。底层命令仅供技术参考。

【虎牙直播源】前端逆向实战:JS解析直播地址参数与加密逻辑

1. 从浏览器抓取到逆向解析:我的虎牙直播源探索之路 大家好,我是老张,一个在AI和大模型领域摸爬滚打了十多年的技术老兵。最近业余时间喜欢在虎牙看看游戏直播,有时候想用自己习惯的播放器(比如VLC或者PotPlayer)来观看,却发现官方只提供了网页和客户端两种方式。这让我这个技术控有点手痒——能不能自己拿到那个最原始的直播流地址呢?网上确实能找到不少别人分享的“直播源”,但说实话,这些链接失效得太快了,官方随便更新一下参数或者加密方式,之前的地址就全废了。所以我一直觉得,与其到处找别人给的“鱼”,不如自己学会“渔”的方法。今天我就把自己折腾虎牙直播源的全过程,特别是前端JavaScript逆向解析参数加密逻辑的实战经验,毫无保留地分享给大家。整个过程完全在浏览器端进行,不需要服务器,小白也能跟着操作。我会把每个步骤、遇到的坑以及解决方案都讲清楚,保证你看完就能自己动手搞定。 你可能要问,为什么非要自己解析?直接录屏不行吗?录屏当然可以,但那损失画质、占用资源,而且不够“极客”。我们想要的是那个最原始的、可以被任何标准播放器识别的流媒体地址(通常是M3U8或FLV格式)。这个地址被

AI如何帮你快速找到JXX登录网页最新域名

快速体验 1. 打开 InsCode(快马)平台 https://www.inscode.net 2. 输入框内输入如下内容: 开发一个智能域名追踪系统,能够自动检测JXX登录网页的最新域名变更。系统需要包含以下功能:1. 定时爬取JXX相关页面,检测域名变化;2. 通过DNS解析验证域名有效性;3. 发现新域名后自动通知用户;4. 提供历史域名记录查询。使用Python实现,集成requests库进行网页请求,dnspython库进行DNS解析,并添加邮件通知功能。 1. 点击'项目生成'按钮,等待项目生成完整后预览效果 AI如何帮你快速找到JXX登录网页最新域名 最近在做一个需要频繁访问JXX网站的项目,但发现这个网站的登录域名经常变更,每次都要花时间到处找最新地址,特别影响工作效率。于是研究了下如何用AI辅助开发一个智能域名追踪系统,自动帮我解决这个问题。 系统设计思路 1. 定时爬取检测:系统需要定期自动访问JXX相关页面,检查是否有新域名出现。这里用Python的requests库就能实现,设置合理的请求间隔避免被封禁。 2.