83 行 Ruby 的蓝调即兴生成器
刻板印象的蓝调
如果让我用代码来生成音乐,我可能优先的类型就是爵士。像是流行音乐,要是来几颗不和谐音,这听起来可就一点都不流行了。但爵士不一样,弹错一次是弹错,如果能连续弹错两次,那就是真正的爵士艺术家了(逃)。但爵士对乐理的要求太高,要想完全靠代码把逻辑理清楚那也不是一件容易的事。一个好的爵士即兴实在是太难了,不过我们可以根据一些刻板印象来做一做。我就想到了十二小节蓝调和声进行以及蓝调音阶,单靠这两个就一下子把需要的伴奏和旋律的模式给提供了一个大概,也许可以试试用代码来自动生成看看。
合成器方面我用的是 Sonic Pi,一款基于 Ruby 的实时编码音乐合成器。其实我之前用过一两次,也不算太会用这个库,也是突发奇想然后现学的。
节奏生成
首先还是要有一个节奏生成器,由于是蓝调音乐,节奏其实可以比较随性。我的基本想法很简单,在给定一个小节里随机填充四分、八分音符。
FOURTH = 0.5
EIGHTH = FOURTH / 2
SIXTEENTH = EIGHTH / 2
def generate_rhythms(candidates=[EIGHTH, FOURTH], remaining_time=FOURTH*4)
rhythms = []
while remaining_time > candidates.min
rhythm = choose(candidates)
rhythms << rhythm
remaining_time -= rhythm
end
rhythms << remaining_time if remaining_time > 0
rhythms
end
节奏上想要比较 swing 一点。一个刻板印象的 swing 就是两个八分音符前一拍长一点,后一拍短一点。所以我就在随机填充后在做了一些手动上的修改:
def generate_rhythms(candidates=[EIGHTH, FOURTH], remaining_time=FOURTH*4)
rhythms = []
while remaining_time > candidates.min
rhythm = choose(candidates)
rhythms << rhythm
remaining_time -= rhythm
end
rhythms << remaining_time if remaining_time > 0
# Generate Swing Pattern
(rhythms.length - 1).times do |i|
if rhythms[i] == EIGHTH and rhythms[i + 1] == EIGHTH
rhythms[i] = EIGHTH + SIXTEENTH
rhythms[i + 1] = EIGHTH - SIXTEENTH
end
end
rhythms
end
写完节奏,我们写一个测试来听一下。
live_loop :rhythm_test do
use_synth :pulse
rhythms = generate_rhythms
rhythms.each do |t|
play :C3, release: t
sleep t
end
end
感觉还行。
贝斯
有了节奏我们接下来就可以写贝斯。贝斯反正很多人听不见,也不用花很大力气写。
十二小节蓝调的和弦基本上就是四个一级,两个四级两个一级,一个五级一个四级再两个一级如此重复。贝斯我的想法就是写一个 walking bass(走路贝斯)。基本上就是第一个音是和弦根音(否则听起来就不像这个和弦了),后面的音就是在九和弦的三度、七度、九度音上随便爬就行。
CHORDS_BASE = [:C2, :C2, :C2, :C2, :F2, :F2, :C2, :C2, :G2, :F2, :C2, :C2]
live_loop :bass_line do
use_synth :fm
CHORDS_BASE.each do |c|
notes = chord(c, '9').to_a
rhythms = generate_rhythms
play notes[0], release: rhythms[0]
sleep rhythms[0]
rhythms[1..-1].each do |t|
play choose([notes[1], notes[3], notes[4]]), release: t
sleep t
end
end
end
感觉差不多。
左手伴奏
写完了贝斯,最好在高一个八度再写一个左手伴奏,节奏可以和贝斯错开。因为已经有贝斯了,这个伴奏我不想要有和弦根音。然后节奏上希望有一些错落感,不要像 bass 一样从头弹到尾。于是我加了一个随机变量,当到达当前节奏时 50% 演奏和弦,50% 概率直接跳过。
CHORDS_HARMONY = [:C3, :C3, :C3, :C3, :F3, :F3, :C3, :C3, :G3, :F3, :C3, :C3]
live_loop :chord do
use_synth :fm
CHORDS_HARMONY.each do |c|
notes = chord(c, '9').to_a
rhythms = generate_rhythms
rhythms.each do |t|
play [notes[1], notes[3], notes[4]], release: t if choose([true, false])
sleep t
end
end
end
听起来是这样的。这蓝调的味道一下子就来了。
即兴
布鲁斯的即兴旋律其实和写中国五声音阶的即兴旋律差不多。是一个上上下下的循环,然后在里面再做一些变化。我这里选用的音阶是 C Eb F Gb G Bb 和 C1,然后两个小节一生成。默认根据其在小节中的大概位置生成一个上升或下降的旋律,然后提供一个 offset 从 -2 到 2 之间随机。在原先旋律上加上一个 offset,如果超出了我们的音程就 clip 掉。这样应该就能产生一个整体上有上下,但是实际上又有很多变化的旋律了。
和贝斯以及和弦不太一样的是,为了有更多的变化感,节奏要从二分、四分、八分、十六分里任意随机。然后被跳过的概率不用 50% 那么大,10% 应该就可以。
BLUES_SCALE = [:C4, :Eb4, :F4, :Gb4, :G, :Bb4, :C5]
melody_reverse = false
live_loop :melody do
use_synth :beep
rhythm = generate_rhythms([FOURTH*2, FOURTH, EIGHTH, SIXTEENTH], remaining_time=FOURTH*8)
scale_notes = BLUES_SCALE
scale_notes = scale_notes.reverse if melody_reverse
melody_reverse = !melody_reverse
rhythm.each_with_index do |t, i|
offset = choose([-2, -1, 1, 2])
new_index = (i.to_f / rhythm.length * scale_notes.length).to_i + offset
new_index = 0 if new_index < 0
new_index = scale_notes.length - 1 if new_index >= scale_notes.length
note = scale_notes[new_index]
play note, release: t if rrand(0, 1) < 0.9
sleep t
end
end
最后我们得到了这样的成品:
总结
完整代码如下:
CHORDS_BASE = [:C2, :C2, :C2, :C2, :F2, :F2, :C2, :C2, :G2, :F2, :C2, :C2]
CHORDS_HARMONY = [:C3, :C3, :C3, :C3, :F3, :F3, :C3, :C3, :G3, :F3, :C3, :C3]
BLUES_SCALE = [:C4, :Eb4, :F4, :Gb4, :G, :Bb4, :C5]
FOURTH = 0.5
EIGHTH = FOURTH / 2
SIXTEENTH = EIGHTH / 2
def generate_rhythms(candidates=[EIGHTH, FOURTH], remaining_time=FOURTH*4)
rhythms = []
while remaining_time > candidates.min
rhythm = choose(candidates)
rhythms << rhythm
remaining_time -= rhythm
end
rhythms << remaining_time if remaining_time > 0
# Generate Swing Pattern
(rhythms.length - 1).times do |i|
if rhythms[i] == EIGHTH and rhythms[i + 1] == EIGHTH
rhythms[i] = EIGHTH + SIXTEENTH
rhythms[i + 1] = EIGHTH - SIXTEENTH
end
end
rhythms
end
live_loop :bass_line do
use_synth :fm
CHORDS_BASE.each do |c|
notes = chord(c, '9').to_a
rhythms = generate_rhythms
play notes[0], release: rhythms[0]
sleep rhythms[0]
rhythms[1..-1].each do |t|
play choose([notes[1], notes[3], notes[4]]), release: t
sleep t
end
end
end
live_loop :chord do
use_synth :fm
CHORDS_HARMONY.each do |c|
notes = chord(c, '9').to_a
rhythms = generate_rhythms
rhythms.each do |t|
play [notes[1], notes[3], notes[4]], release: t if choose([true, false])
sleep t
end
end
end
melody_reverse = false
live_loop :melody do
use_synth :beep
rhythm = generate_rhythms([FOURTH * 2, FOURTH, EIGHTH, SIXTEENTH], remaining_time=FOURTH*8)
scale_notes = BLUES_SCALE
scale_notes = scale_notes.reverse if melody_reverse
melody_reverse = !melody_reverse
rhythm.each_with_index do |t, i|
offset = choose([-2, -1, 1, 2])
new_index = (i.to_f / rhythm.length * scale_notes.length).to_i + offset
new_index = 0 if new_index < 0
new_index = scale_notes.length - 1 if new_index >= scale_notes.length
note = scale_notes[new_index]
play note, release: t if rrand(0, 1) < 0.9
sleep t
end
end
一共 83 行,好像还挺像那么回事的。而且由于是一个 Loop,放着程序可以无穷无尽地生成下去,作为什么免费无版权的背景音乐素材好像还是个挺不错的来源。