MIDI音乐设备音频单元:播放两个相同音高的音符,停止其中一个



说到AudioUnits,我还是个新手,所以如果我的问题很基本,请原谅我。

我正在使用MusicDevice AudioUnit播放一些音符。我正在使用MusicDeviceMIDIEvent发送备忘开启和备忘关闭信息。它运行良好。有时应该同时发出多个音符,所以我可能会连续发送两个音符。有时这些音符恰好有相同的音高。然后,当我想关闭其中一个音符时,我会为这个音高发送一个音符关闭事件。但此消息会关闭音高的所有音符。当然,这种行为很有道理,但我想问一下人们通常是如何处理这个问题的。

我应该使用不同的通道同时进行音符吗?或者手动管理音符,比如用一个计数集来保存当前正在播放的音高,并且只有在音高的最后一个实例停止播放后才发送音符关闭事件?或者完全是别的什么?

编辑:

由于这是在iOS上,我必须使用kAudioUnitSubType_Sampler作为AudioUnitsubType。虽然文档只提到这种类型是单声道的,但我现在怀疑它也是单声道的。这当然可以解释这种行为。我仍然想知道,如果我真的有一种复调乐器,我会怎么做。

编辑2:

我做了更多的测试,现在在我看来,在任何频道上发送音符关闭消息都会停止所有频道上相同音高的所有音符。我在http://developer.apple.com/library/ios/#samplecode/LoadPresetDemo/Introduction/Intro.html并修改了stopPlay[低/中/高]音符方法,以在某个随机频道上发送音符关闭消息(如果您必须知道的话,分别在频道7、8和9上)。尽管消息上的注释是在通道0上发送的,但它仍然会停止注释。这是预期的行为吗?

为了确保我没有犯愚蠢的错误,以下是发送备忘和备忘的方法:

- (IBAction) startPlayLowNote:(id)sender {
UInt32 noteNum = kLowNote;
UInt32 onVelocity = 127;
UInt32 noteCommand =    kMIDIMessage_NoteOn << 4 | 0;
OSStatus result = noErr;
require_noerr (result = MusicDeviceMIDIEvent (self.samplerUnit, noteCommand, noteNum, onVelocity, 0), logTheError);
logTheError:
if (result != noErr) NSLog (@"Unable to start playing the low note. Error code: %d '%.4s'n", (int) result, (const char *)&result);
}

- (IBAction) stopPlayLowNote:(id)sender {
//note the channel!
UInt32 noteNum = kLowNote;
UInt32 noteCommand =    kMIDIMessage_NoteOff << 4 | 7;
OSStatus result = noErr;
require_noerr (result = MusicDeviceMIDIEvent (self.samplerUnit, noteCommand, noteNum, 0, 0), logTheError);
logTheError:
if (result != noErr) NSLog (@"Unable to stop playing the low note. Error code: %d '%.4s'n", (int) result, (const char *)&result);
}

我很确定,在同一频道上发生两个相同音高的音符开启事件后,音符关闭的行为是未定义的。有些乐器可能同时关闭两个音符,有些乐器可能关闭其中一个音符,需要关闭第二个音符才能关闭另一个音符。

如果你真的需要同时有两个相同音高的音符,它们应该在不同的通道上。

编辑已发布代码

我在你的链接中尝试了这个示例项目,并以你在发布代码中所做的相同方式更改了频道。事实证明kAudioUnitSubType_Sampler确实是单声道的,所以它忽略了MIDI通道参数。因此,如果你想用kAudioUnitSubType_Sampler同时有两个音高相同的音符,你必须创建两个单独的实例。

注意,kAudioUnitSubType_Sampler不是单声道的。它是复调的,因为它可以同时演奏多个音高。

使用MusicDevice.h的MusicDeviceStartNote()MusicDeviceStopNote()怎么样?它为音符使用一个唯一的标记,所以你应该能够区分两个相同音高的音符。

相关内容

最新更新