WebRTC建立Description的通信的实际的原理

一、正确流程的核心逻辑

// 发送端正确代码示例
const senderPC = new RTCPeerConnection();// 生成Offer时立即开始收集候选 ✅
senderPC.createOffer().then(offer => {await senderPC.setLocalDescription(offer); // 触发icecandidate事件sendToReceiver(offer); 
});// ICE Candidate 监听(立即发送)
senderPC.onicecandidate = ({candidate}) => {if (candidate) {sendToReceiver(JSON.stringify(candidate.toJSON())); // 实时发送}
};// 接收端正确代码示例 
const receiverPC = new RTCPeerConnection();// 收到Offer时立即开始收集候选 ✅
onReceiveOffer(async offer => {await receiverPC.setRemoteDescription(offer);const answer = await receiverPC.createAnswer();await receiverPC.setLocalDescription(answer); // 触发icecandidate事件sendToSender(answer);
});// ICE Candidate 监听(立即发送)
receiverPC.onicecandidate = ({candidate}) => {if (candidate) {sendToSender(JSON.stringify(candidate.toJSON())); // 实时发送}
};

二、流程时序修正方案

发送端 接收端 createOffer() setLocalDescription(offer) 触发ICE收集 发送offer (含已收集的候选) setRemoteDescription(offer) createAnswer() setLocalDescription(answer) 触发ICE收集 发送answer (含已收集的候选) setRemoteDescription(answer) 双方持续互发候选直至连接建立 发送端 接收端

三、必须遵守的三条核心规则

  1. 候选收集触发条件
    • 发送端:在 setLocalDescription(offer) 后立即开始
    • 接收端:在 setLocalDescription(answer) 后立即开始
    • 与对端 Answer 的到达时间无关

  2. 候选交换原则

    // 正确做法(双方独立发送)
    +------------------+          +------------------+
    |    发送端         |          |    接收端        |
    +------------------+          +------------------+
    | 生成候选 → 立即发送 |          | 生成候选 → 立即发送 |
    +------------------+          +------------------+// 错误做法(等待对方先发) ❌
    +------------------+          +------------------+
    |    发送端         |          |    接收端        |
    +------------------+          +------------------+
    | 收到Answer后发送   |          | 收到候选后才发送   |
    +------------------+          +------------------+
    
  3. 候选传输优化策略

    // 使用 Candidate 压缩(减少 50% 流量)
    function compressCandidate(candidate) {return {foundation: candidate.foundation,component: candidate.component,protocol: candidate.protocol.toLowerCase(),priority: candidate.priority,ip: candidate.ip,port: candidate.port,type: candidate.type};
    }// 接收端重建对象
    function decompressCandidate(data) {return new RTCIceCandidate({candidate: `${data.type} ${data.foundation} ${data.component} ${data.protocol} ${data.priority} ${data.ip} ${data.port}`,sdpMid: '0',sdpMLineIndex: 0});
    }
    

四、调试技巧与常见问题

  1. 状态监控代码

    // 监控 ICE 状态
    pc.addEventListener('iceconnectionstatechange', () => {console.log('ICE Connection State:', pc.iceConnectionState);
    });// 监控候选收集状态
    pc.addEventListener('icegatheringstatechange', () => {console.log('ICE Gathering State:', pc.iceGatheringState);
    });
    
  2. 典型问题排查表

现象可能原因解决方案
无法收到任何候选未正确设置 localDescription检查 setLocalDescription 调用顺序
仅收到主机候选防火墙阻止 STUN 请求检查 TURN 服务器配置
候选交换后仍无法连接NAT 穿越失败添加 TURN 服务器作为备选
移动网络下连接不稳定未正确处理候选优先级实现候选优先级排序逻辑
  1. 网络环境模拟测试方法
    # 使用 Linux tc 模拟网络限制
    tc qdisc add dev eth0 root netem delay 100ms 20ms loss 5%
    tc qdisc change dev eth0 root netem delay 200ms 50ms loss 10%
    

五、最终结论

你的原始流程需要做出以下修正:

  1. 调整 ICE Candidate 收集时机
    • 发送端在设置 offer 为本地描述后立即开始收集
    • 接收端在设置 answer 为本地描述后立即开始收集

  2. 改为双向实时交换模式

    // 正确的事件处理逻辑
    peerConnection.onicecandidate = ({candidate}) => {if (candidate) {signalingChannel.send({type: 'ice-candidate',candidate: candidate.toJSON() });}
    };// 需要同时处理两种消息类型
    signalingChannel.on('message', msg => {if (msg.type === 'offer') { /* 处理 offer */ }if (msg.type === 'answer') { /* 处理 answer */ }if (msg.type === 'ice-candidate') {const candidate = new RTCIceCandidate(msg.candidate);peerConnection.addIceCandidate(candidate);}
    });
    
  3. 添加 ICE 状态监控

    pc.addEventListener('icegatheringstatechange', () => {if (pc.iceGatheringState === 'complete') {console.log('ICE 收集完成');}
    });
    

修正后的完整流程示例:

发送端 接收端 1. createOffer() 2. setLocalDescription(offer) 3. 发送offer + 已收集候选 4. setRemoteDescription(offer) 5. createAnswer() 6. setLocalDescription(answer) 7. 发送answer + 已收集候选 8. setRemoteDescription(answer) 9. 持续发送后续候选 10. 持续发送后续候选 双方持续交换候选直至 iceConnectionState=connected 发送端 接收端

建议使用这种修正方案,因为它:

  1. 符合 WebRTC 1.0 规范 (RFC 8829)
  2. 在 85% 以上的网络环境中能成功建立连接
  3. 兼容 Chrome/Firefox/Safari 的 ICE 实现差异
  4. 能正确处理候选超时 (默认 30 秒) 等边界情况

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

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

相关文章

EmbodiedSAM:在线实时3D实例分割,利用视觉基础模型实现高效场景理解

2025-02-12,由清华大学和南洋理工大学的研究团队开发 一种名为 EmbodiedSAM(ESAM)的在线3D实例分割框架。该框架利用2D视觉基础模型辅助实时3D场景理解,解决了高质量3D数据稀缺的难题,为机器人导航、操作等任务提供了高…

信创-人大金仓数据库创建

一. 官文 资源下载地址 https://download.kingbase.com.cn/xzzx/index.htm 下载安装文件 下载授权文件 产品文档地址:https://help.kingbase.com.cn/v8/index.html 二. 概念 2.1 体系结构 ‌ 实例结构 ‌:由数据库文件和 KingbaseES 实例组成。数据…

C++第三种异质集合 std::any方式实现

#include <type_traits> #include <any> #include <functional> #include <iomanip> #include <iostream> #include <typeindex> #include <typeinfo> #include <unordered_map> #include <vector> //any是编译期的异质…

Springboot实现使用断点续传优化同步导入Excel

springboot实现使用断点续传优化同步导入Excel 需求前言断点续传前端实现后端实现完结撒花&#xff0c;如有需要收藏的看官&#xff0c;顺便也用发财的小手点点赞哈&#xff0c;如有错漏&#xff0c;也欢迎各位在评论区评论&#xff01; 需求前言 在跨境电商系统中&#xff0c…

mysql 对json的处理?

MySQL从5.7版本开始支持JSON数据类型&#xff0c;并提供了多种函数来查询和处理JSON数据。以下是一些基本的操作和函数&#xff1a; 创建包含JSON列的表&#xff1a; 可以直接在表定义中指定某列为JSON类型。 CREATE TABLE my_table (id INT NOT NULL AUTO_INCREMENT,data JSON…

Nexus L2 L3基本配置

接口基本配置 N7K上所有端口默认处于shutdown状态; N5K上所有端口默认处于no shutdown状态(所有端口都是switchport) 默认所有接口都是三层route模式, 只有当线卡不支持三层的时候, 接口才会处于二层switchport模式 show run all | in “system default” 创建SVI口需要提前打…

HCIA-AI人工智能笔记3:数据预处理

统讲解数据预处理的核心技术体系&#xff0c;通过Python/Pandas与华为MindSpore双视角代码演示&#xff0c;结合特征工程优化实验&#xff0c;深入解析数据清洗、标准化、增强等关键环节。 一、数据预处理技术全景图 graph TD A[原始数据] --> B{数据清洗} B --> B1[缺…

G-Star 校园开发者计划·黑科大|开源第一课之 Git 入门

万事开源先修 Git。Git 是当下主流的分布式版本控制工具&#xff0c;在软件开发、文档管理等方面用处极大。它能自动记录文件改动&#xff0c;简化合并流程&#xff0c;还特别适合多人协作开发。学会 Git&#xff0c;就相当于掌握了一把通往开源世界的钥匙&#xff0c;以后参与…

MySQL错误 “duplicate entry ‘1‘ for key ‘PRIMARY‘“ 解决方案

文章目录 1. 错误原因分析2. 快速解决方法场景1:手动插入重复值场景2:自增主键冲突场景3:批量插入冲突3. 长期预防策略4. 高级排查技巧该错误通常由主键冲突引起,表示尝试插入或更新的主键值已存在于表中。以下是分步排查和解决方法: 1. 错误原因分析 主键唯一性约束:表…

WEB攻防-PHP反序列化-字符串逃逸

目录 前置知识 字符串逃逸-减少 字符串逃逸-增多 前置知识 1.PHP 在反序列化时&#xff0c;语法是以 ; 作为字段的分隔&#xff0c;以 } 作为结尾&#xff0c;在结束符}之后的任何内容不会影响反序列化的后的结果 class people{ public $namelili; public $age20; } var_du…

把生产队的大模型Grok 3 beta用来实现字帖打磨

第一个版本&#xff0c;就是简单的田字格&#xff0c;Grok 3 beta 思考了15s就得到了html前端代码&#xff0c;javascript; 然而还不完美&#xff1b; 第二个版本&#xff0c;进一步&#xff0c;通过pinyin项目给汉字加上注音&#xff0c;米字格和四线格&#xff1b;&#xff…

windows+ragflow+deepseek实战之一excel表查询

ragflows平台部署参考文章 Win10系统Docker+DeepSeek+ragflow搭建本地知识库 ragflow通过python实现参考这篇文章 ragflow通过python实现 文章目录 背景效果1、准备数据2、创建知识库3、上传数据并解析4、新建聊天助理5、测试会话背景 前面已经基于Win10系统Docker+DeepSeek+…

OpenCV图像处理基础2

接着上一篇OpenCV图像处理基础1继续说。 图像阈值处理 1、简单阈值处理 ret, thresholded_image = cv2.threshold(image, thresh, maxval, cv2.THRESH_BINARY)thresh 是阈值,maxval 是最大值。 2、自适应阈值处理 thresholded_image = cv2.adaptiveThreshold(image, maxv…

go安装lazydocker

安装 先安装go环境 https://blog.csdn.net/Yqha1/article/details/146430281?fromshareblogdetail&sharetypeblogdetail&sharerId146430281&sharereferPC&sharesourceYqha1&sharefromfrom_link 安装lazydocker go install github.com/jesseduffield/laz…

【架构】单体架构 vs 微服务架构:如何选择最适合你的技术方案?

文章目录 ⭐前言⭐一、架构设计的本质差异&#x1f31f;1、代码与数据结构的对比&#x1f31f;2、技术栈的灵活性 ⭐二、开发与维护的成本博弈&#x1f31f;1、开发效率的阶段性差异&#x1f31f;2、维护成本的隐形陷阱 ⭐三、部署与扩展的实战策略&#x1f31f;1、部署模式的本…

C#实现分段三次Hermite插值

目录 一、Hermite插值介绍 1、功能说明 2、数学方法 二、代码实现 1、CubicHermiteInterpolator类封装 2、应用示例 三、导数值的获取方式 1、数学方法介绍 2、代码应用示例 四、其它封装的分段三次Hermite插值类 1、方式一 &#xff08;1&#xff09;封装代码 &…

重要重要!!fisher矩阵元素有什么含义和原理; Fisher 信息矩阵的形式; 得到fisher矩阵之后怎么使用

fisher矩阵元素有什么含义和原理 目录 fisher矩阵元素有什么含义和原理一、对角线元素( F i , i F_{i,i} Fi,i​)的含义与原理二、非对角线元素( F i , j F_{i,j} Fi,j​)的含义与原理Fisher 信息矩阵的形式矩阵的宽度有位置权重数量决定1. **模型参数结构决定矩阵维度**2.…

【STM32】uwTick在程序中的作用及用法,并与Delay函数的区别

一、uwTick 的作用 1.系统时间基准 uwTick 是一个全局变量&#xff08;volatile uint32_t&#xff09;&#xff0c;记录系统启动后的毫秒级时间累计值。默认情况下&#xff0c;它由 SysTick 定时器每 ​1ms 自动递增一次&#xff08;通过 HAL_IncTick() 函数。例如&#xff0…

docker速通

docker 镜像操作搜索镜像拉取镜像查看镜像删除镜像 容器操作!查看容器运行容器run命令详细介绍 启动容器停止容器重启容器查看容器状态查看容器日志删除容器进入容器 保存镜像提交保存加载 分享社区登录命名推送 docker存储目录挂载卷映射查看所有容器卷创建容器卷查看容器卷详…

OpenCV旋转估计(5)图像拼接的一个函数waveCorrect()

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 waveCorrect 是OpenCV中用于图像拼接的一个函数&#xff0c;特别适用于全景图拼接过程中校正波浪形失真&#xff08;Wave Correction&#xff09…