查看“︁區塊摺積”︁的源代码
←
區塊摺積
跳转到导航
跳转到搜索
因为以下原因,您没有权限编辑该页面:
您请求的操作仅限属于该用户组的用户执行:
用户
您可以查看和复制此页面的源代码。
{{No footnotes|time=2022-06-15T13:32:57+00:00}} '''區塊折積''',又稱'''區塊卷積'''、'''分段折積'''、'''分段卷積'''、'''分塊卷積'''等,是一種計算線性離散[[卷积]]的方法,在[[信号处理]]以及[[线性时不变系统理论|線性非時變系統]]領域的應用層面上有相當高的價值。 區塊卷積主要常使用於計算一固定[[离散信号]]與另一非固定離散訊號間的線性卷積,且非固定訊號的長度通常較固定方長上許多,一個很具體的例子就是計算一訊號經有限長度[[滤波器设计|濾波器]]後的輸出。 其主要具有兩個優點,第一,其[[計算複雜性理論|計算複雜度]]較低,在輸入訊號長度為 <math> N </math>時,區塊卷積的[[計算複雜性理論|計算複雜度]]為<math>\Omicron(N)\!</math>,而直接對整段輸入訊號進行卷積計算的複雜度為<math>\Omicron(N \log N)\!</math>。 第二,當我們要以硬體進行實作時,區塊卷積僅需要單一固定的硬體架構,即可處理不同長度(甚至是無窮長)的輸入,可以像工廠的[[裝配線|流水生產線]]一般連續的接受輸入並同時連續的吐出結果, 而如果要直接對整段輸入訊號進行卷積,則需要對不同的輸入長度設計不同的硬體運算架構,在應用上是不切實際並且沒有必要的。 目前對於區塊卷積,主流使用的英文專有詞彙為sectioned convolution,但在一些[[漢語圈|中文圈]]作者所撰寫的文章中,也可能會採用取「塊」字翻英後的block convolution。 == 計算 == 區塊卷積的計算核心概念,便是將較長的訊號進行分段,將原先的求卷積問題分解為規模較小的問題,再將結果進行整合,以得到原先問題的結果。 但根據將訊號分段以及將結果整合的方式,可以再將區塊卷積細分為兩個類別,可见下文说明。 ===卷積的性質觀察=== 首先,在對區塊卷積的計算法進行細分之前,我們需要先了解一些關於[[卷积|卷積]]的性質,以及關於切分後子問題的結果與母問題結果的關聯。 假設有一長度為<math> N </math>的離散訊號(通常很長): <math> x[n] ( n = 0 , 1 , \dots , (N-1) ) </math> 與一長度為<math> M </math>的離散訊號(通常很短): <math> h[n] ( n = 0 , 1 , \dots , (M-1) ) </math> 將兩者進行卷積後的結果<math> y[n] = x[n] * h[n]</math>,我們可以知道其長度應該為<math> (N+M-1) </math>: <math> y[n] ( n = 0 , 1 , \dots , (N+M-2) ) </math> 根據卷積的定義,我們可以知道其中<math> y </math>的第<math> n' </math>點輸出<math> y[n'] </math>, 會受到相對應位置的<math> x[n'] </math>,以及其向前<math>(M-1) </math>點的影響: <math> y[n'] = \sum_{m=0}^{M-1} x[n'- m] h[m] = x[n'- (M-1)]h[M-1]+ x[n' - (M-2)]h[M-2] + \dots + x[n']h[0]</math> 這時如果我們將<math> x[n] </math>切出長度為<math> L </math>的小段,並假設起始位置為<math> n_0 </math>: <math> x_L[n] = x[n] ( n = n_0 , n_0+1 , \dots , n_0+(L-1) ) </math> 我們可以發現,其與<math> h[n] </math>進行卷積後的結果<math> y_L[n] = x_L[n] * h[n]</math>,和原先母問題在對應區段的結果<math> y[n] </math>存在差異, 具體來說,<math> y_L[n]</math>的前段部分少考量了<math> x[n] </math>中<math> n<n_0 </math>部分的影響,例如在<math> y_L[n]</math>第一點<math> y_L[n_0]</math> 就僅考量一項<math> x </math>的影響,和<math> y[n_0] </math>相差許多: <math> y_L[n_0] = x[n_0]h[0] \neq y[n_0] = x[n_0- (M-1)]h[M-1]+ \dots + x[n_0]h[0]</math> 同理,在<math> y_L </math>的後段部分也會有類似的問題,少考量了<math> x </math>在段落之後的影響,而只有計算到部分殘留影響的結果。 只有在<math> y_L </math>的中段部分,會與母問題結果<math> y </math>完全相同。 而要如何處理這個差異,並透過多段的<math> y_L </math>還原出<math> y </math>,便是以下兩種方法的最主要區別。 ===重疊-儲存之摺積法=== 重疊-儲存算法,便是捨棄子結果<math> y_L </math>的前後段,只取中段與母結果<math> y </math>相同的部分並進行拼接的演算法。 具體來說,重疊-儲存算法切分問題的主要考量為母結果<math> y </math>, 在實作這個算法的時候必須先決定將母結果<math> y </math>切分的長度。 假設決定將母結果切分為長度<math> L_y </math>的區間, 根據前面的討論,我們可以得知: 1.<math> x_L </math>的長度需要為<math> (L_y + (M-1)) </math>才能夠確保<math> L_y </math>長度的有效區段 2.在每次長度為<math> (L_y + 2(M-1)) </math>的計算結果<math> y_L </math>中,需要捨去前後各<math> (M-1) </math>長度的部分 3.在第一個區段,因為<math> x[n] </math>在<math> n<0 </math>的區域是沒有資訊的,所以<math> x_{L0} </math>的長度只需要<math> L_y </math>即可 綜合以上三點,我們可以寫出所需要的切分區段描述: <math> x_{L0}[n] = x[n] (n \in 0 \sim (L_y - 1)) </math> <math> x_{Lk}[n] = x[n] (n \in (kL_y - (M-1) ) \sim ((k+1)L_y - 1) , k \in 1 \sim (\left \lceil \frac{M+N-1}{L_y} \right \rceil - 1) ) </math> 以及結果的描述: <math> y[n] = y_{Lk}[n] (n \in kL_y \sim ((k+1)L_y - 1) , k \in 0 \sim (\left \lceil \frac{M+N-1}{L_y} \right \rceil - 1)) </math> 而重疊-儲存算法便是得名於其在輸入區段的切分上會有重疊的部分,須將前一個子問題的輸入後半先行儲存再作為下一個子問題的輸入前半。 這個算法的好處在於不需要額外的加法,並且如果在子問題求解時是採取直接計算卷積的方式,可以不用在訊號後面補零。 但缺點為同樣的分段數量下,每個子問題會需要處理較長的訊號,子問題計算量可能會較大一些。 ===重疊-相加之摺積法=== 重疊-相加算法,則是將子結果<math> y_L </math>的後段, 與下一段子結果<math> y_L </math>的前段, 像拼圖一樣根據對應的編號進行相加, 重建出完整母結果<math> y </math>的演算法。 具體來說,重疊-儲存算法切分問題的主要考量為輸入<math> x </math>, 在實作這個算法的時候必須先決定將輸入<math> x </math>切分的長度。 假設決定將輸入切分為長度<math> L </math>的區間, 我們可以直接寫出所需要的切分區段描述: <math> x_{Lk}[n] = x[n] (n \in kL \sim ((k+1)L - 1) , k \in 0 \sim (\left \lceil \frac{N}{L} \right \rceil - 1)) </math> 以及結果的描述: <math> y[n] = x[n]*h[n] = \sum_{k=0}^{(\left \lceil \frac{N}{L} \right \rceil - 1)} x_{Lk}[n]*h[n] = \sum_{k=0}^{(\left \lceil \frac{N}{L} \right \rceil - 1)} y_{Lk}[n] </math> 這個算法特別需要注意的便是,重建出的母結果<math> y </math>牽涉到子結果對應項目的相加,要正確地對上才會是正確的結果,例如<math> y[L+1] </math>就會是兩個子結果的相加: <math> y[L+1] = \underbrace{ x[(L+1) - (M-1)]h[M+1] + \dots + x[(L+1) - 2]h[2] }_{y_L0[L+1]} + \underbrace{ x[(L+1) - 1]h[1] + x[(L+1) - 0]h[0] }_{y_L1[L+1]} </math> 而重疊-相加算法便是得名於其在輸出的子結果上會有項目編號重疊的部分,須將其進行相加才能得到母結果。 這個算法的好處在於切分的方式相當直覺,而且在同樣的分段數量下,每個子問題需要處理的訊號長度較短,子問題計算量較小。 但缺點為可能需要一些額外的加法,而且就算採取直接計算卷積的方式來計算子問題,也仍需要補零在訊號後方。 == 複雜度與比較 == 雖然區塊卷積具有兩種差異不小的算法,但總體來說,對於每個固定架構的子問題,所需要的計算量都是常數, 而子問題的數量則正比於輸入訊號長度<math> N </math>,所以計算複雜度對兩個算法來說都是<math>\Omicron(N)\!</math>, 相較於使用[[快速傅里叶变换]]技巧進行整個問題的計算複雜度<math>\Omicron(N \log N)\!</math>要來的低。 所以一般來說在<math> N </math>很大時,顯然的使用區塊卷積會較處理整個問題來的有效率。 不過,有時候輸入訊號<math> x </math>並不具有足夠的長度可以保證區塊卷積較有效率, 又或者在一些極端情況下,直接按定義進行計算卷積可能會更有效率, 所以接著便是要稍微談論一下不同主流計算卷積方法的計算量, 以及如何根據輸入訊號長度<math> N </math>與固定訊號長度<math> M </math>間的關係選擇算法。 ===直接計算卷積=== 如果是根據卷積的定義,直接以乘法及加法計算卷積的話, 我們可以知道,每一個<math> x[n] </math>都需要輪流和每一個<math> h[n] </math>進行相乘, 所以總共需要的乘法數量為 <math> N \times M </math> 又因為訊號可能是複數,所以上述的乘法為複數乘法,將其統一以時數乘法進行實作後可以得到需要的實數乘法數為 <math> 3N \times M </math> 可以發現,其實直接計算卷積對於輸入長度<math> N </math>來說,其複雜度也是<math>\Omicron(N)\!</math>, 但其常數為<math> 3M </math>,當<math> M </math>增大時,運算量會增加很快,運算效率會不及區塊卷積, 但相對的當<math> M </math>落在很小的值時,直接計算卷積可能會是較高效率的計算方法。 ===使用快速傅立葉變換技巧計算卷積=== 除了直接計算卷積之外,另外一種主流的卷積計算方法,便是透過[[卷积定理]], 將卷積的計算化為兩次[[傅里叶变换|傅立葉變換]]相乘後再進行反傅立葉變換的問題, 實作如下: <math> y[n] = x[n]*h[n] = IFFT_P(FFT_P(x[n]) \times FFT_P(h[n])) </math> 其中<math>FFT_P</math>為<math>P</math>點快速傅立葉變換, <math>IFFT_P</math>為逆變換(形式與快速傅立葉變換相同),<math>P</math>根據圓周卷積定理必須大於<math>(M+N-1)</math>, 而<math> x[n] </math>及<math> h[n] </math>需透過補零來補齊長度。 如果採用這個方法,並且假設<math> h[n] </math>為固定不變,可以將<math> FFT_P(h[n]) </math>計算一次後儲存備用, 那可以得到所需乘法數為: <math>2 MUL_P + 3P</math> 其中<math>MUL_P</math>為<math>P</math>點快速傅立葉變換所需的乘法數, 估計為<math> \frac{3}{2} P \log_2 P</math>, 乘兩倍是因為需計算<math> FFT_P(x[n]) </math>與最後的逆變換, <math>3P</math>則是因為<math>FFT_P(x[n]) \times FFT_P(h[n]) </math>含有<math>P</math>個複數的乘法,統一轉換為實數乘法則是<math>3P</math>。 將<math> P=(M+N-1) \approx (M+N)</math>與<math>MUL_P</math>的估計值帶入上面的所需乘法數進行估計,可以得到所需乘法數為: <math> 3(M+N)(\log_2 (M+N) + 1) \approx 3(M+N)\log_2 (M+N) </math> 可以發現,使用快速傅立葉變換技巧計算卷積對於輸入長度<math> N </math>來說,其複雜度是<math>\Omicron(N \log N)\!</math>, 但當<math> M </math>大約與<math> N </math>落在接近的數量級時,運算量會較其他方法低上許多。 ===使用區塊卷積搭配快速傅立葉算法計算卷積=== 最後,另外一種主流的卷積計算法便是先利用區塊卷積將問題拆分,再將拆分後的問題透過[[快速傅里叶变换|快速傅立葉算法]]來處理, 而之所以通常不是搭配直接計算的方式, 是因為我們通常會假設拆分過後的問題中,要進行卷積的兩訊號長度會差不多長, 而根據前面的討論,我們已經知道使用快速傅立葉轉換來進行計算會有效率的多。 而使用區塊卷積搭配快速傅立葉算法計算卷積的複雜度, 和前面兩種方法的計算複雜度只與<math>M , N</math>有關不同, 分段所使用的段落長度<math>L</math>也會很大程度的影響這個方法的計算量。 以重疊-相加算法作為代表, 假設採用的分段長度為<math>L</math>, 可以得到分段的段數<math>S</math>為: <math>S = \left \lceil \frac{N}{L} \right \rceil \approx \frac{N}{L} </math> 而每個子問題的實際計算量根據上面可得: <math>2 MUL_P + 3 P</math> 乘上需要解的子問題數可得實際總計算量: <math>2S \times MUL_P + 3S \times P</math> 同樣根據上面可再進一步估計得: <math> \frac{N}{L} 3(M+L-1)(\log_2 (M+L-1) + 1) \approx \frac{N}{L} 3(M+L)\log_2 (M+L) </math> 可以發現這個計算法的複雜度為<math>\Omicron(N)\!</math>, 並且常數約為<math>\frac{3(M+L)}{L} \log_2 (M+L) </math>, 在<math>M</math>的數值略大時會較採取直接計算會較有效率, 但在<math>M</math>與<math>N</math>的數量級接近時, 因為<math>M</math>的大小被迫推升, 導致比起直接對整段訊號進行快速傅立葉算法的表現還要差。 所以根據上述可以估計這個方法可以展現優勢的區段, 大概是界在前面兩種方法的<math>M , N</math>比例之間。 而在實際應用時,因為快速傅立葉轉換的計算量在局部會有不小的波動,並不如理論值估計那般平滑,所以為了得到較好的計算效率, 除了須將上面的計算量視作<math>L</math>的函數,透過數值繪圖的方法或微分,求得在理論估計上最適合的<math>L</math>值之外, 還需在求得的最適合子問題大小附近進行找尋, 挑選數個可能的合適子問題傅立葉轉換大小, 再將反求得的分段長度帶回計算運算量, 以求得真正的最佳分段長度<math>L</math>。 例如:在 <math>N = 2000,M=17</math> 的情況下,求得的理論最適合分段長度為 <math>L_0 = 85</math> 理論最適合子問題傅立葉轉換大小為 <math>P_0 = L_0 + M -1 = 101</math>點 但在<math>72</math>點傅立葉轉換的計算量有一個明顯的低谷,將由此逆推得到的可能分段長度 <math>L = 72 - M + 1 = 56</math> 與其他可能值都帶回去計算實際計算量後,可以確認<math>56</math>才是最佳的問題分段長度。 最後,同樣是基於快速傅立葉轉換的計算量在局部的波動,在極為特殊的情況下,如果<math>M \leq 4</math>, 我們便有可能使子問題僅需使用<math>4</math>點傅立葉轉換,而在這樣的情況下雖會使用大量的加法, 但可不必在傅立葉轉換上使用任何乘法,可能會較直接計算卷積有效率。 == 參考資料 == {{Reflist|2}} Jian-Jiun Ding, [http://djj.ee.ntu.edu.tw/ADSP.htm advanced digital signal processing lecture note] {{Wayback|url=http://djj.ee.ntu.edu.tw/ADSP.htm |date=20200508102040 }}, the Department of Electrical Engineering, National Taiwan University (NTU), Taipei, Taiwan, 2022. == 外部連結 == [[:重疊-儲存之摺積法]] [[:重疊-相加之摺積法]] [[Category:電機工程]]
该页面使用的模板:
Template:No footnotes
(
查看源代码
)
Template:Reflist
(
查看源代码
)
Template:Wayback
(
查看源代码
)
返回
區塊摺積
。
导航菜单
个人工具
登录
命名空间
页面
讨论
不转换
查看
阅读
查看源代码
查看历史
更多
搜索
导航
首页
最近更改
随机页面
MediaWiki帮助
特殊页面
工具
链入页面
相关更改
页面信息