-1

我有一个 uint32_t 如下:

uint32_t midiData=0x9FCC00;

我需要将这个 uint32_t 分成更小的部分,以便 9 成为它自己的实体,F 成为它自己的实体,CC 成为它自己的实体。如果您想知道我在做什么,我正在尝试分解 MIDI 消息的各个部分,以便在我的程序中更容易管理它们。

我找到了这个解决方案,但问题是我不知道如何将它应用到 CC 部分,而且我不确定这个方法是否适用于 C++。

这是我到目前为止所拥有的:

uint32_t midiData=0x9FCC00;

uint32_t status = 0x0FFFFF & midiData; // Retrieve 9
uint32_t channel = (0xF0FFFF & midiData)>>4; //Retrieve F
uint32_t note = (0xFF00FF & midiData) >> 8; //Retrieve CC

这对 C++ 是否正确?我问的原因是因为我以前从未使用过 C++,它使用 > 和 < 的语法总是让我感到困惑(因此我倾向于避免使用它)。

4

3 回答 3

1

您也可以在 C++ 中使用位移位运算符>>和位掩码运算符。&但是,如何使用它存在一些问题:

运算符v1 & v2给出一个由在两者中设置的位构建的数字v1and v2,例如,0x12 & 0xF0给出0x10, not 0x02。此外,位移运算符采用位数,十六进制数中的单个数字(通常称为半字节)由 4 位组成(0x0..0xF 需要 4 位)。所以,如果你有0x12并且想要得到0x01,你必须写0x12 >>4。因此,您的班次也需要调整:

#define BITS_OF_A_NIBBLE 4

unsigned char status = (midiData & 0x00F00000) >> (5*BITS_OF_A_NIBBLE);
unsigned char channel = (midiData & 0x000F0000) >> (4*BITS_OF_A_NIBBLE);
unsigned char note = (midiData & 0x0000FF00) >> (2*BITS_OF_A_NIBBLE);
unsigned char theRest = (midiData & 0x000000FF);
于 2017-04-30T21:48:37.553 回答
0

在某种程度上,你有它倒退。

在布尔逻辑中(& 是按位与),与 0 将排除它。知道F在十六进制中是1111二进制的,像这样的0x9FCC00 & 0x0FFFFF行将为您提供除 9 之外的所有十六进制数字,这与您想要的相反。

因此,对于状态:

uint32_t status = 0xF000000 & midiData; // Retrieve 9

实际上,这会给你 0x900000。如果你想要 0x9(十进制也是 9),你需要将结果移位。

现在,右移位运算符(例如,X >> 4)意味着将 X 向右移动 4 位;除以 16。那是 4 位,而不是 4 个十六进制数字。1 个十六进制数字 == 4 位,因此要从 0x900000 获得 9,您需要0x900000 >> 20.

因此,将它们放在一起,得到状态 9:

uint32_t status = (0xF000000 & midiData) >> 20;

类似的过程将为您提供所需的剩余值。

于 2017-04-30T21:37:35.377 回答
0

一般来说,我建议先换档,然后再戴口罩 - 它不太容易出错:

uint8_t cmd = (midiData >> 16) & 0xff;
uint8_t note = (midiData >> 8) & 0x7f;     // MSB can't be set
uint8_t velocity = (midiData >> 0) & 0x7f; // ditto

然后拆分cmd变量:

uint8_t status = (cmd & 0xf0);             // range 0x00 .. 0xf0
uint8_t channel = (cmd & 0x0f);            // range 0 .. 15

我个人不会费心将status值映射回范围 0 .. 15 - 人们普遍认为 eg0x90是“注释”,而不是普通值9

于 2017-05-24T21:47:58.570 回答