空间音频空间音频能够将真实世界的声音体验带到虚拟数字世界,为用户提供更具吸引力的在线体验。通过声网提供的空间音频技术,你可以在虚拟互动场景下模拟声音在现实环境中的传播特点:
超拟真空间塑造效果
利用范围音频、声音模糊、空气衰减模拟等技术,完美模拟现实听觉感受。
当声源和其他角色之间的距离、方位或朝向发生变化时,不同的角色听到的声音大小和方向均不相同。
当声源和其他角色之间的距离超过设置的范围后,后者无法听到声源发出的声音。
通过设置隔声区域、空气衰减、声音模糊等音效,完美模拟现实的听觉感受。
3D 高保真
根据面部朝向、音源朝向、3D 空间内音源的相对位置进行音效处理渲染。
支持 48 kHz 全频带采样和 3D 高保真音效处理及渲染。
多平台适配
支持 iOS、Android、macOS、Windows、Web、Unity、Flutter、React Native、Electron、Unreal 等平台。
超低延时、低功耗、低成本
空间音频算法采用先进的前端处理模式,并通过云服务同步空间坐标,端云协同处理模式能够有效降低整体延时和功耗。
相比传统的左右声道立体声,声网提供的空间音频技术在声音的立体感和空间感上都有明显的提升,两者区别如下所示:
对比项传统立体声声网空间音频立体感左右两个维度以世界坐标系的 x、y、z 坐标轴表示右、上、前三个维度空间感通过调节左右声道的音量,体现具有空间感的声音使用空间音频算法,通过设置距离、方位、朝向等参数,塑造声音的空间感用户体验平面立体、自然、真实
适用场景
社交语聊
在语聊房 App 界面中,以九宫格的形式展示所有用户头像,并对每个用户赋予一个坐标和朝向。在互动过程中,你可以感受到周围的用户发出的声音大小和方位跟其所在的位置有关。如果你在屏幕中拖动自己的头像,随着你跟某个用户之间的距离逐渐变远,你会听到该用户的声音也越来越小,超出一定范围后会彻底听不到,符合现实环境中声音的传播特点。
游戏 & 元宇宙
在游戏和元宇宙这类 3D 场景中,你可以使用空间音频技术实现更多功能,如下表所示:
功能描述声音模糊对指定远端用户或媒体播放器开启声音模糊处理。
例如:在咖啡厅场景中,如果需要制造两个虚拟角色 “说悄悄话” 的效果,可以对他们开启声音模糊处理模式,其他角色会听到被处理过的、模糊的声音。
范围音频根据当前场景的大小设置一个音频接收范围,在该范围以内,声音传播得越远,接收方听到的声音越小。你可以通过设置声音衰减系数,在不同的场景下获得更加拟真的衰减效果:
在丛林场景中,将衰减系数设置为 0.9,声音在传播时迅速衰减。
在平原场景中,将衰减系数设置为 0.1,声音可以传播得很远。
超拟真空间塑造为全方位模拟真实场景的空间感,对每个虚拟角色赋予 4 组坐标,分别表示以下维度:
虚拟角色空间坐标(x,y,z):虚拟角色在 3D 空间里的坐标。
虚拟角色面部朝向坐标(x,y,z):虚拟角色在场景中的面部朝向坐标。
虚拟角色头顶朝向坐标(x,y,z):虚拟角色的头顶指向的坐标,可结合面部朝向坐标确定虚拟角色的动作和姿态。例如:虚拟角色平躺的时候,面部朝向天空,头顶朝向前方。
右手朝向(x,y,z):直角坐标系是使用左手坐标系还是右手坐标系。在一个场景中只能使用一种坐标系。
隔声区域定义一个隔声区域并对隔声区域内的音源设置声音衰减系数,使区域外的接收方听区域内的音源时,会体验到类似真实环境中声音在遇到建筑隔断时的衰减效果。
例如:在 KTV 场景中,当虚拟角色在 KTV 包厢外时,只能听到来自包厢内的较小的声音;在打开包厢门的时候,会在一瞬间体验到包厢里唱歌的声音迎面扑来。
在线会议
在线会议场景下,你可以将所有上麦用户分布到主持人的周围,并赋予每个用户坐标。用户听到的每个声音都有方向感和空间感,模拟真实环境下的会议室体验,相比传统在线会议,能够带给用户全新的与会体验、降低用户疲惫感。
技术原理
实现用户空间音频和媒体播放器空间音频时的原理有所不同:
用户空间音频:自行部署服务器,用于发送本地空间位置以及接收远端用户的空间位置,然后由 SDK 计算用户间的相对位置,从而计算出相应的空间音频参数。
媒体播放器空间音频:通过 SDK 计算本地用户和媒体播放器的相对位置,然后计算出相应的空间音频参数。
前提条件
在进行操作之前,请确保你已经在项目中实现了基本的实时音视频功能。详见实现音视频互动。
实现空间音频
本节介绍如何使用 SDK 实现空间音频,API 调用时序如下图所示。
初始化空间音频引擎
调用 getLocalSpatialAudioEngine 和 initialize 初始化 ILocalSpatialAudioEngine 对象,开启空间音频。
TypeScriptrtcEngine = createAgoraRtcEngine();localSpatial = rtcEngine.getLocalSpatialAudioEngine()localSpatial.initialize();
设置音频属性和场景
调用 setAudioProfile,将 profile 设置为你预期的音频编码属性。
调用 setAudioScenario,将 scenario 设置为 AudioScenarioGameStreaming,以获得预期的音质效果。
TypeScriptrtcEngine.setAudioProfile(AudioProfileType.AudioProfileSpeechStandard);rtcEngine.setAudioScenario(AudioScenarioType.AudioScenarioGameStreaming);
设置音频接收范围
调用 setMaxAudioRecvCount 设置音频接收范围内最多可接收的音频流数,建议 maxCount 取值 ≤ 16。
调用 setAudioRecvRange 设置可接收音频的最大范围,单位为米,建议 range 取值 > 0。
TypeScriptlocalSpatial.setMaxAudioRecvCount(2);localSpatial.setAudioRecvRange(50);
更新空间位置
在用户空间音频场景下,调用 updateSelfPosition 和 updateRemotePosition 方法,分别更新本地用户和远端用户的位置。
在媒体播放器空间音频场景下,调用 updateSelfPosition 和 updatePlayerPositionInfo 方法,分别更新本地用户和媒体播放器的位置。
推荐的调用时机如下所示:
当频道内有新用户加入。
当本地、远端用户或媒体播放器的相对位置发生改变。
根据实际需求调用。
TypeScript// 更新本地用户位置localSpatial.updateSelfPosition( [0, 0, 0], // position [1, 0, 0], // axisForward [0, 1, 0], // axisRight [0, 0, 1] // axisUp);// 更新远端用户位置localSpatial.updateRemotePosition( uid, // 远端用户 ID { position: [0, 0, 0], axisForward: [1, 0, 0], });// 更新媒体播放器位置localSpatial.updatePlayerPositionInfo( mediaPlayer.getMediaPlayerId(), // 媒体播放器 ID { position: [0, 0, 0], axisForward: [1, 0, 0], });
设置空间音频参数
调用 setRemoteUserSpatialAudioParams 或 setSpatialAudioParams,针对远端用户或媒体播放器设置空间音频参数。如果要实现特定音效,可参考如下参数设置:
空气衰减效果:将 enable_air_absorb 设置为 true;并将 speaker_attenuation 设置为预期的声音衰减系数。
声音模糊效果:将 enable_blur 设置为 true。
TypeScriptrtcEngine.setRemoteUserSpatialAudioParams( uid, { enable_air_absorb: true, speaker_attenuation: 0.5, enable_blur: true, });
(可选)设置隔声区域
调用 setZones 设置一个隔声区域和声音衰减系数。当音源(可以为用户或媒体播放器)跟接收方分属于隔声区域区域内部和外部时,会体验到类似真实环境中声音在遇到建筑隔断时的衰减效果。
(可选)调用 setRemoteAudioAttenuation 或 setPlayerAttenuation 分别针对用户和媒体播放器设置声音衰减属性,并指定是否使用该设置强制覆盖 setZones 中的声音衰减系数。
TypeScriptlocalSpatial.setZones( [ { zoneSetId: 1, // 隔声区域的 ID position: [0, 0, 0], // 隔声区域的空间中心点 forward: [1, 0, 0], // 以 position 为起点,向前的单位向量 right: [0, 1, 0], // 以 position 为起点,向前的单位向量 up: [0, 0, 1], // 以 position 为起点,向前的单位向量 forwardLength: 10, // 隔声区域的前向长度 rightLength: 10, // 隔声区域的右向长度 upLength: 10, // 隔声区域的上向长度 audioAttenuation: 0.5, // 隔声区域和外部互通时的声音衰减系数 } ], 1);localSpatial.setRemoteAudioAttenuation( mediaPlayer.getMediaPlayerId(), // 媒体播放器 ID 0.5, false);localSpatial.setPlayerAttenuation( mediaPlayer.getMediaPlayerId(), // 媒体播放器 ID 0.5, false);
(可选)设置耳机均衡效果
调用 setHeadphoneEQPreset 方法,选择预设的耳机均衡器收听音频,以达到预期的音频体验。
(可选)如果执行上一步后仍未达到预期,你可以调用 setHeadphoneEQParameters 自行调节耳机均衡效果,执行该方法后,setHeadphoneEQPreset 方法设置的预设值会被覆盖。
TypeScriptrtcEngine.setHeadphoneEQPreset(HeadphoneEqualizerPreset.HeadphoneEqualizerOverear);rtcEngine.setHeadphoneEQParameters(10, 10);
关闭空间音频
在互动过程中,你可选择暂停或关闭空间音频。
暂停指定远端用户在本地的空间音频
如果不想继续体验远端用户在本地的空间音频、或远端用户已退出频道,调用 removeRemotePosition 删除该用户的空间位置信息,以节省计算资源。
注意当远端用户离开频道时,必须调用 removeRemotePosition 删除该用户的空间位置信息,否则可能导致本地听不到来自其他远端用户的空间音频。
(可选)如果之后想恢复该用户在本地的空间音频,可调用 updateRemotePosition 方法重新设置远端用户的位置信息。
TypeScript// 暂停指定远端用户在本地的空间音频localSpatial.removeRemotePosition(uid)// 恢复指定远端用户在本地的空间音频localSpatial.updateRemotePosition( uid, // 远端用户 ID { position: [0, 0, 0], axisForward: [1, 0, 0], });
暂停所有远端用户在本地的空间音频
如果你不想再继续体验本地的空间音频,可以调用 clearRemotePositions 删除所有远端用户的空间位置信息。
注意调用该方法会导致本地听不到所有远端用户的音频,声网建议你谨慎调用该方法。
(可选)如果之后还需要听到远端用户的音频,你需要重新调用 ILocalSpatialAudioEngine 中的 muteAllRemoteAudioStreams(false) 恢复订阅远端用户的音频流。
TypeScript// 删除所有远端用户的空间位置信息localSpatial.clearRemotePositions();// 恢复订阅远端用户的音频流localSpatial.muteAllRemoteAudioStreams(false);
暂停本地用户在远端的空间音频
调用 ILocalSpatialAudioEngine 中的 muteLocalAudioStream(true),取消发布本地音频流。
(可选)如果之后想再次开启空间音频,可再次调用该方法并将参数设置为 false,即可恢复空间音频。
TypeScript// 取消发布本地音频流localSpatial.muteLocalAudioStream(true);// 恢复发布本地音频流localSpatial.muteLocalAudioStream(false);
关闭空间音频
在频道内调用 RtcEngine 中的 enableSpatialAudio 并将参数设置为 false,此时所有跟空间音频相关的设置都会被重置。
(可选)如果之后想再次开启空间音频,可再次调用该方法并将参数设置为 true,然后重新调用相关 API 设置空间音频效果。
调用 ILocalSpatialAudioEngine 中的 release 销毁 ILocalSpatialAudioEngine 对象。
TypeScript// 关闭空间音频rtcEngine.enableSpatialAudio(false);// 销毁 ILocalSpatialAudioEngine 对象localSpatial.release();
参考信息
示例项目
声网提供了开源的空间音频示例项目供你参考,你可以前往下载或查看其中的源代码。
信息示例项目中的 SpatialAudio.tsx 和 LocalSpatialAudioEngine.tsx 需配合使用才能实现空间音频。
SpatialAudioSpatialAudioLocalSpatialAudioEngineLocalSpatialAudioEngine
API 参考
ILocalSpatialAudioEngine
initialize
setMaxAudioRecvCount
setAudioRecvRange
updateSelfPosition
updateRemotePosition
updatePlayerPositionInfo
setRemoteUserSpatialAudioParams
setSpatialAudioParams
setZones
setRemoteAudioAttenuation
setPlayerAttenuation
clearRemotePositions
muteLocalAudioStream
muteAllRemoteAudioStreams
enableSpatialAudio
destroy
RtcEngine
setHeadphoneEQPreset
setHeadphoneEQParameters