国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 開(kāi)發(fā) > Python > 正文

用Python制作簡(jiǎn)單的鋼琴程序的教程

2024-09-09 19:02:23
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

錄一段音頻,把它的音高改變50次并把每一個(gè)新的音頻匹配到鍵盤(pán)的一個(gè)鍵位,你就能把電腦變成一架鋼琴!

一段音頻可以被編碼為一組數(shù)值的數(shù)組(或者列表),像這樣:

201541171623980.png (512×112)

我們可以在數(shù)組中每隔一秒拿掉一秒的值來(lái)將這段音頻的速度變成兩倍。

201541171709750.png (512×112)

如此我們不僅將音頻的長(zhǎng)度減半了,而且我們還將它的頻率翻倍了,這樣使得它擁有比原來(lái)更高的音高(pitch)。

相反地,假如我們將數(shù)組中每個(gè)值重復(fù)一次,我們將得到一段更慢,周期更長(zhǎng),即音高更低的音頻:

201541171802247.png (512×112)

這里提供一個(gè)可以按任意系數(shù)改變音頻速度的任意簡(jiǎn)單的Python函數(shù):
 

import numpy as np def speedx(sound_array, factor):  """ 將音頻速度乘以任意系數(shù)`factor` """  indices = np.round( np.arange(0, len(snd_array), factor) )  indices = indices[indices < len(snd_array)].astype(int)  return sound_array[ indices.astype(int) ]

這個(gè)問(wèn)題更困難的地方在于改變音頻長(zhǎng)度的同時(shí)保持它的音高(變速,音頻拉伸(sound stretching)),或者在改變音頻的音高的同時(shí)保持它的長(zhǎng)度(變調(diào)(pitch shifting))。
變速

變速可以通過(guò)傳統(tǒng)的相位聲碼器(phase vocoder,感興趣的朋友可以讀一下維基百科的頁(yè)面)來(lái)實(shí)現(xiàn)。首先將音頻分解成重疊的比特,然后將這些比特重新排列使得他們重疊得更多(將縮短聲音的長(zhǎng)度)或者更少(將拉伸音頻的長(zhǎng)度),如下圖所示:

201541171835595.png (300×102)

困難之處在于重新排列的比特可能很?chē)?yán)重的互相影響,那么這里就需要用到相位變換來(lái)確保它們之間沒(méi)有影響。這里有一段Python代碼,取自這個(gè)網(wǎng)頁(yè)(打不開(kāi)的話,您懂的。――譯者注):
 

def stretch(sound_array, f, window_size, h):  """ 將音頻按系數(shù)`f`拉伸 """   phase = np.zeros(window_size)  hanning_window = np.hanning(window_size)  result = np.zeros( len(sound_array) /f + window_size)   for i in np.arange(0, len(sound_array)-(window_size+h), h*f):     # 兩個(gè)可能互相重疊的子數(shù)列    a1 = sound_array[i: i + window_size]    a2 = sound_array[i + h: i + window_size + h]     # 按第一個(gè)數(shù)列重新同步第二個(gè)數(shù)列    s1 = np.fft.fft(hanning_window * a1)    s2 = np.fft.fft(hanning_window * a2)    phase = (phase + np.angle(s2/s1)) % 2*np.pi    a2_rephased = np.fft.ifft(np.abs(s2)*np.exp(1j*phase))     # 加入到結(jié)果中    i2 = int(i/f)    result[i2 : i2 + window_size] += hanning_window*a2_rephased   result = ((2**(16-4)) * result/result.max()) # 歸一化 (16bit)   return result.astype('int16')

 
變調(diào)

一旦你實(shí)現(xiàn)了變速以后,變調(diào)就不難了。如果需要一個(gè)更高的音高,可以先將這段音頻拉伸并保持音高不變,然后再加快它的速度,如此最后得到的音頻將具有原始音頻同樣的長(zhǎng)度,更高的頻率,即更高的音高。

把一段音頻的頻率翻倍將把音高提高一個(gè)八度,也就是12個(gè)半音。因此,要將音高提高n個(gè)半音的話,我們需要將頻率乘上系數(shù)2^(n/12):

發(fā)表評(píng)論 共有條評(píng)論
用戶(hù)名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 开江县| 乌拉特中旗| 马山县| 内丘县| 永寿县| 林西县| 洪江市| 林芝县| 恭城| 永吉县| 吉隆县| 建水县| 涪陵区| 桐梓县| 荆州市| 中阳县| 阿克陶县| 成安县| 洪湖市| 孝昌县| 拉萨市| 南安市| 邵武市| 白朗县| 万源市| 津市市| 赤峰市| 进贤县| 南川市| 兴业县| 枝江市| 松江区| 宽城| 龙南县| 扶沟县| 江孜县| 阳江市| 平南县| 江城| 四会市| 泸州市|