07|如何通过算法自动快速地消除回声?

前面几节,我们讲了实时音频前处理中的降噪算法。从这节开始,我们将会看看语音交互或者说音频通信领域的另一项不可或缺的技术:声学回声消除(Acoustic Echo Cancellation, 简称 AEC)。

回声消除算法在实时音频互动链路中和很多其它模块以及硬件都会有耦合。这节我们主要看看实时音频互动链路中回声是如何产生的以及回声消除算法的基本原理。

需要注意的是,这节课涉及到的公式比较多。不过不用担心,只要你理解了原理,就很容易能明白公式的含义,从而能够记住公式的定义。

回声产生的原因

回声是如何产生的呢?可以通过下面的图来看一下,这是一个出现回声的经典场景。

图1 回声产生的原理

图 1 中 B 端的人在说话,说话的声音会被 B 端的麦克风采集。麦克风采集到的语音信号转为数字信号后经过网络传输到 A 端,在 A 端的扬声器会把收到的语音信号转成声波播放出来,同时 A 端的麦克风又会把扬声器的声音采集回来,通过网络又传回给了 B 端。这时 B 端的人就会听到自己发出去的声音,把这个声音就叫做回声。

在音频实时互动的场景里,回声会严重影响通话体验,自己的声音不断被打断,而且对端的声音可能会和回声混在一起,这样会显著降低语音的可懂度。那么相对地,AEC 的任务就是为了消除这个回声。在 AB 端互通的时候,需要使用 AEC,在 A 端把麦克风采集到的信号中和 B 端相关的声音去掉,只保留 A 端的音源的声音发给 B 端。

那么是不是把 A 端麦克风采集到的信号减去 A 端扬声器中的信号就能把回声消除了呢?先来看看 A 端的回声消除的基本原理,再回过头来解答这个问题。

回声消除的基本原理

在近端接收到的远端的声音信号我们把它叫做参考信号 x(n)。在经过扬声器播放、空气传播、房间墙体反射、麦克风采集后,参考信号不可避免地会产生很多变化。把这个变化用数学的方式来表达就叫做回声路径的传递函数,一般记作 f。那么近端麦克风采集的回声信号echo(n) 可以用公式 1 来表示:

近端除了回声信号还有近端自己的声音,比如近端人说话的声音。那么近端麦克风收到的信号 z(n) 其实是近端声音 y(n) 和回声信号之和,如公式 2 所示:

回声消除算法的目的就是通过算法估计出回声路径的传递函数f,把算法估计出的传递函数定义为 f’, 那么经过回声消除后得到的近端信号 z’(n) 为公式 3 所示:

将公式 1 和公式 2 代入公式 3,可以得到公式 4:

可以看到如果估计的传递函数 f’ 和真实的传递函数 f 是一致的,那么回声就被完美消除了。但在真实的场景中传递函数的估计是一件比较困难的事情。

这是因为 AEC 算法需要面对复杂的、时变的声学环境。比如,扬声器和麦克风的播放失真、采集失真会给声学信号带来很多非线性的变化,并且设备、系统调度的不稳定可能造成回声和远端接收信号的延迟抖动。同时诸如房间的混响、设备所处位置的变化,都会带来回声路径的变化。

因此,AEC 算法必须能够快速地自适应地去估算出这些回声路径的变化。如果估计不准,就会导致回声泄漏或者近端声音被压制,甚至造成丢字、卡顿等现象,从而严重影响实时音频互动的质量。所以回到之前的问题,我们知道回声消除是做减法,但又不是直接相减就能解决。

那么是如何让算法自动快速地进行回声消除呢?AEC 技术经过半个多世纪的发展,在不断的实践中已经摸索出一套以自适应滤波为基础的回声消除方法。自适应滤波的核心思想就是用实时更新的滤波器的系数来模拟真实场景的回声路径,然后结合远端信号来估计出回声信号,再从近端采集的混合信号中减去估计的回声,从而达到消除回声的目的。

自适应滤波器

接下来先来看看自适应滤波的基本原理。

维纳滤波

在一个相对稳定的声学环境中,回声路径中的延迟和房间的混响、音量大小的变化其实都可以看作是对远端信号做了一系列的线性变化。这种线性变化可以用一个线性离散的 FIR 线性滤波器来表示,公式 1 就变成了公式 5:

其中 wk​ 代表第 k 个滤波器系数。如果在近端除了回声信号没有别的声音的时候,那么其实接收到的信号就是回声信号,即 z(n)=echo(n)。这种情况一般叫做“单讲”。在这种情况下,我们的回声估计误差 e(n) 可用公式 6 表示:

之前降噪讲到过的降噪算法第三招中的维纳滤波。其实维纳滤波就是以估计误差 e(n) 的最小平方作为最优解的线性滤波器。也就是通过计算最小均方差(Mean Square Error,简称 MSE)来求取滤波器系数 W(wk​∈W)。公式 7 为求解最小 MSE 的代价函数。

我们知道想要让函数值最小,其实就是让函数的全微分等于 0。其求解过程可以用维纳 - 霍夫方程来表示:

其中,R 是参考信号 x(n) 序列的相关矩阵,P 是参考信号和回声信号 echo(n) 的互相关矢量。这样滤波器的系数似乎就可以得到了。但是试想一下,假设音频的采样率是 48kHz,如果只取 1 秒的信号来求解,那么 R 矩阵的维度就是 48000 乘以 48000。

显然要实时求一个这么大的矩阵的逆矩阵,算力是不可能支持实时计算的。把这种直接求得的解叫做维纳解,虽然它是最准确的,但是计算量过于庞大,而且当回声路径变化的时候需要重新计算维纳解。所以很显然维纳解并不适合在实时音频互动中使用

那么有什么办法能实时求解滤波器系数呢?其实自适应滤波器的核心思想是在面对回声路径不断变化的场景,比如移动电话等时,可以使用梯度下降法来迭代的计算滤波器系数。

在计算代价函数的时候让 W 系数朝着梯度相反的方向或者说朝着减少代价函数的方向移动。随着迭代次数的增加,W 会逐渐的向维纳解收敛。这样当回声路径发生变化的时候,W 就会重新收敛,从而就可以实时的追踪回声路径的变化了。

那么这个迭代计算具体是怎么实现的呢?下面就通过两个算法来看一下迭代计算的过程。

LMS、NLSM 算法

最小均方算法 LMS(Least Mean Square)是最早提出,也是最基础的自适应滤波方法。它的基本原理可以表示为公式 9:

其中,W(n) 代表第 n 次迭代时的滤波器的系数向量,X(n) 是第 n 次迭代的输入向量,e(n) 是第 n 次迭代的误差,μ 是步长因子。可以看到步长因子 μ 决定了滤波器系数的收敛速度,且 μ 越大收敛越快。

最小均方算法的梯度下降是随机的,随着迭代次数的增加它会不断逼近维纳解。但是看到公式 9 里梯度下降也会受到输入向量 X(n) 大小的影响。也就是说,如果远端信号音量比较小,那这时系数向量的收敛速度会变得很慢;反过来,X(n) 很大的时候会导致梯度放大,从而系数向量的收敛变快。

那么怎么解决这个音量变化带来的收敛波动问题呢?

其实可以通过 X(n) 的大小来动态调节步长因子,这样就可以把 X(n) 进行归一化。这就是 NLMS 算法的由来。NLMS 算法的迭代步骤如公式 10、11 所示:

其中,μ~​ 是一个常量,取值范围在 0~2,δ 为一个大于 0 的常数,主要是为了防止 X(n) 过小导致的梯度爆炸。NLMS 相对于 LMS 通过归一化的方式提升了算法的收敛速度。目前 NLMS 算法已经成为 AEC 算法中最常用的算法之一。

线性滤波器的挑战和解决方法

那么只有 NLMS 是不是就足够了呢?其实这里面还有三个很重要的问题没有解决。下面就来简单介绍一下这三个问题。

延迟估计

第一个问题是回声延迟。公式 9~11 中 X(n) 是一段有限长度的输入信号,这个长度也就是常说的滤波器的感知长度。如果实际回声信号的传递路径很长,比如有很大的延迟和混响,那么就需要用一个很长的 X(n) 作为输入才能估计出回声信号的传递函数。然而一个感知长度很大的滤波器需要的算力也会随之增加,这样就会对 AEC 的实时性造成挑战。

为了解决这个问题,最先想到的就是把延迟进行单独计算。可以看到假设回声信号的延迟为 dn,那么在公式 5 里延迟的表示就是 wk​=0,k∈[0,dn−1]。如果能够把延迟估计出来,那么权重为 0 的系数就不需要放到 NLMS 里去估计了,那么整体的算力就可以降下来。同时有了延迟估计,NLMS 只需要估计后面非 0 部分的权重,从而收敛速度也可以变快。

延迟估计的方法也比较简单,其实就是移动远端信号的起始位置,然后和回声信号计算互相关性,并找到互相关最大的位置。这个位置就是我们要的延迟。

双讲检测

第二个问题是双讲。前面讲维纳滤波的时候讲到了“单讲”。所谓“双讲”,就是远端和近端同时说话或者说两侧都有明显的声音。那在这个时候麦克风采集的信号除了回声还有混入了近端的声音。又 NLMS 是依赖于回声信号来进行估计的,而这时如果用麦克风采集的信号作为回声信号,就会导致滤波器无法收敛到正确的位置,从而产生回声泄漏或近端声音被损伤。

因此,一般会利用远端和近端信号先做一个简单的判断,此时是单讲还是双讲状态。如果是单讲,那么滤波器系数照常迭代更新;如果是双讲,则需要通过调节步长因子等方法停止或者减缓滤波器的更新。双讲检测的方法主要是结合能量和远、近端信号的相干性来做一个判断。如果远端和近端能量都比较高但是相干性却不强,那么就说明远端和近端都有声音,也就是双讲的状态。

这里可以思考一下,在现实生活中可能经常会碰到的一个现象:如果和对方打网络电话的时候,我们从一个房间走到另一个房间,比如从会议室走到走廊,对面反馈说听到了回声。

这其实就和 AEC 的双讲时的策略有关,如果你和对端同时说话恰巧在此刻你换了个地方,也就是回声路径发生了改变。但由于是双讲的状态,滤波器没有及时更新,这时候就会漏回声。所以双讲检测可以防止滤波器发散。但这其实也并不是一个完美的解决方案,可能还会导致回声泄漏。只是这种双讲时,恰巧换房间的情况不是那么常见,所以双讲检测依然是回声策略中常见的调整依据。

非线性

第三个问题是看到 NLMS 等算法中实际上估计的是一个线性的滤波器。但是之前有讲到扬声器、麦克风等都可能会导致一些非线性的变换。那么这时线性滤波器可能就无法处理了。一般来说一些廉价或者说声学特性比较差的设备导致的非线性失真比较多,所以出现回声的概率也更大。

在实时音频互动刚开始的时候,其实大部分厂商都还是只有线性的回声消除。但现在一般会在线性回声处理之后再集联一个非线性处理,来解决这些线形处理后的残留回声。非线性建模需要兼顾不同设备、环境是一件很有挑战的事情。

除了传统算法,最近几年也有很多通过机器学习的方式来解决非线性的方案,并起到了比较好的效果。究其复杂性,非线性这块将会在下一次中再继续展开来聊聊。

小结

由于采集和播放设备的耦合,在实时音频互动领域,回声消除是实时音频链路中重要的一环。常见的回声消除流程包括双讲检测、延迟估计、线性回声消除、非线性回声消除等步骤。

图2 回声消除的基本步骤

回声消除发展了几十年,依然还是一个比较热门的研究领域。究其原因还是因为它的复杂性,设备、环境、工程部署的实时性甚至是其它的音频模块都可能会对回声消除的效果产生影响。一般把回声消除模块放在紧挨着音频采集模块的位置。也就是说,做完了 AEC 再做降噪、增益调整等其它的音频模块。这样可以尽量减少音频处理对回声路径的复杂性的增加。

回声消除算法其实是在已知一个音源信号的条件下,在多音源混合的音频中消除这一音源。所以有的时候回声消除也被用来做一些音源分离的事情。比如一首歌你已经有伴奏的情况下,对人声和伴奏混合在一起的歌曲,用回声消除就可以提取到清唱(也就是没有伴奏的纯人声)。

  • 3
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值