0

私のシステムでは、WAV ファイルの配列のバイトを使用して単一の WAV ファイルを作成する必要があります。現在、Okio を使用してバッファのデータを読み書きし、最終ファイルにデータを書き込みます。

私はこのドキュメントとこのスタックオーバーフローの質問に従っています:

と...

  • おきお:1.10.0
  • コトリン: 1.0.2-1
  • ジャワ:1.7

私はこのコードを作成しました:

fun mixAudios() {

    try {

        //Create the file used to storage the mixed audio file.
        val file = File(directory, finalFileName)

        //Open the buffer for this file.
        val bufferedSink = Okio.buffer(Okio.appendingSink(file))

        //Data header of the file.
        val header = Buffer()

        //Data of the file.
        val data = Buffer()

        //Do a action for every audio.
        audios.forEach {

            //Try to read the file, if success, return the file.
            Okio.buffer(Okio.source(File(it.address)))?.let { file ->

                //Create a new buffer for every audio address.
                val buffer = Buffer()

                //Read every byte on the buffer.
                file.readAll(buffer)

                //Remove the first 44 items of the buffer.
                buffer.readByteArray(44)

                //Get the buffer and write every byte on the sink.
                data.writeAll(buffer)

                //Close the sink.
                buffer.close()
                file.close()
            }
        }

        //Count of bytes on the data buffer.
        val fileSize = data.size().toInt()

        //The data is ready to be written on the sink.
        data.close()

        val totalFileSize = fileSize + 36
        val byteRate = (SAMPLE_RATE * CHANNELS * BYTES_PER_SAMPLE) / 8

        //Write the header of the final file.
        header.writeUtf8("RIFF")

            //Write the total file size (with the header)
            .writeByte(totalFileSize and 0xff)
            .writeByte((totalFileSize shr 8) and 0xff)
            .writeByte((totalFileSize shr 16) and 0xff)
            .writeByte((totalFileSize shr 24) and 0xff)
    //      .writeIntLe(fileSize) //Inform the size of the chunk, including the header.

            .writeUtf8("WAVE") //Inform the type of file.
            .writeUtf8("fmt ") //Add the "fmt" letters
            .writeIntLe(samplingRate) //fmt chunk

            .writeByte(AUDIO_FORMAT_PCM) //This byte represents the audio format (PCM).
            .writeByte(0)

            .writeByte(CHANNELS) //This byte represents the channels of the audio.
            .writeByte(0)

            //Write the sample rate
            .writeByte(SAMPLE_RATE and 0xff)
            .writeByte((SAMPLE_RATE shr 8) and 0xff)
            .writeByte((SAMPLE_RATE shr 16) and 0xff)
            .writeByte((SAMPLE_RATE shr 24) and 0xff)
    //      .writeIntLe(SAMPLE_RATE) //The sample rate of the audio

            //Write the byte rate
            .writeByte(byteRate and 0xff)
            .writeByte((byteRate shr 8) and 0xff)
            .writeByte((byteRate shr 16) and 0xff)
            .writeByte((byteRate shr 24) and 0xff)
    //      .writeIntLe((SAMPLE_RATE * CHANNELS * BYTES_PER_SAMPLE) / 8) //Byte rate

            .writeByte(CHANNELS * BYTES_PER_SAMPLE / 8) //Block align
            .writeByte(0)

            .writeByte(BYTES_PER_SAMPLE) //Bytes per sample
            .writeByte(0)

            .writeUtf8("data") //File content size

            .writeByte(fileSize and 0xff)
            .writeByte((fileSize shr 8) and 0xff)
            .writeByte((fileSize shr 16) and 0xff)
            .writeByte((fileSize shr 24) and 0xff)
    //      .writeIntLe(fileSize)

            .close()

        with (bufferedSink) {
            writeAll(header)
            writeAll(data)
            close() //Close and write the file on the memory.
        }

        //Do the rest...

        } catch (e: Exception) {
        if (debugEnabled) {
            e.printStackTrace()
        }
      }
    }

ファイルは正常に生成されますが、このオーディオを任意のメディア プレーヤーで開こうとすると、破損しているように見えます。

生成されたこのオーディオ ファイルのバイトを調査しようとすると、結果は次のようになります。

バイト

ヘッダーを正しく書いているかどうかわかりません。この問題を解決するのを手伝ってもらえますか?

ありがとうございました!

4

1 に答える 1

1

samplingRate の値はどこから取得していますか? 80 3E 00 00 (つまり 16000) と書いていますが、実際には 10 00 00 00 (つまり生の PCM フォーマットの場合は 16) のはずです。

ヘッダーのこの時点で、fmt チャンクのサイズを記述する必要があります。

これで問題が解決することを願っています

于 2016-08-31T10:45:01.777 に答える