Мне нужен быстрый способ в языке C # преобразования / приведения массива байтов, кодирующих одно короткое (int16) значение для 2 байтов, в представление с плавающей запятой, как можно быстрее. Производительность узким местом была методом:
samples[sample] = (float)binraryReader.readInt16();
(огромное количество вызовов ввода-вывода, поэтому мне пришлось конвертировать в чтение блоков)
По сути, у меня есть файл, содержащий блок звуковых выборок (~ 100-600 МБ) типа short, тогда, поскольку я могу блокировать только чтение набора байтов, мне нужно создать short из каждой пары байтов, а затем преобразовать это Коротко о представлении с плавающей точкой, так как мне нужно хранить образцы в формате с плавающей точкой.
мой текущий код выглядит примерно так (примерно в 2 раза лучше, чем в предыдущем методе, но все еще слишком долго):
float[] samples = new float[_samplesPerSplit];
byte[] data = new byte[_samplesPerSplit * 2];
for (int c = 0; c < numberOfChunks; c += 1)
{
br.Read(data, 0, _samplesPerSplit * 2);
fixed (byte* bytePtr = data)
{
fixed (float* floatPtr = samples)
{
byte* rPos = bytePtr;
float* fPos = floatPtr;
byte byte0;
byte byte1;
short sampleShort;
for (int sample = 0; sample < _samplesPerSplit; sample += 1)
{
byte1 = *(rPos++);
byte0 = *(rPos++);
// I occasionaly get
// "Negating the minimum value of a
// twos complement number is invalid"
// error if i skip this check, but it slows down
// whole process even more
if (byte0 == 128 && byte1 == 0)
{
sampleShort = 32767;
}
else
{
sampleShort = (short)(((ushort)(byte0)) << 8 | ((ushort)(byte1)));
}
*(fPos++) = (float)sampleShort;
}
}
}
ProcessChunk(samples);
}