在 Android 中使用 MediaCodec 进行 NV21 编码和解码的过程如下:
编码 NV21 数据:
// 创建 MediaCodec 编码器,并配置编码器格式和参数
val encoder = MediaCodec.createEncoderByType("video/avc")
val mediaFormat = MediaFormat.createVideoFormat("video/avc", width, height)
mediaFormat.setInteger(MediaFormat.KEY_BIT_RATE, bitrate)
mediaFormat.setInteger(MediaFormat.KEY_FRAME_RATE, frameRate)
mediaFormat.setInteger(MediaFormat.KEY_COLOR_FORMAT, MediaCodecInfo.CodecCapabilities.COLOR_FormatYUV420SemiPlanar)
mediaFormat.setInteger(MediaFormat.KEY_I_FRAME_INTERVAL, iframeInterval)
encoder.configure(mediaFormat, null, null, MediaCodec.CONFIGURE_FLAG_ENCODE)// 启动编码器
encoder.start()// 创建输入缓冲区和输出缓冲区
val inputBuffers = encoder.inputBuffers
val outputBuffers = encoder.outputBuffers// 将 NV21 数据分割成 Y、U、V 平面
val yPlane = ByteArray(width * height)
val uvPlane = ByteArray(width * height / 2)
System.arraycopy(nv21Data, 0, yPlane, 0, width * height)
System.arraycopy(nv21Data, width * height, uvPlane, 0, width * height / 2)// 编码循环
while (isEncoding) {// 获取空闲的输入缓冲区索引val inputBufferIndex = encoder.dequeueInputBuffer(-1)if (inputBufferIndex >= 0) {val inputBuffer = inputBuffers[inputBufferIndex]inputBuffer.clear()inputBuffer.put(yPlane) // 将 Y 平面数据放入输入缓冲区inputBuffer.position(0)encoder.queueInputBuffer(inputBufferIndex, 0, yPlane.size, presentationTimeUs, 0)presentationTimeUs += 1_000_000 / frameRate // 更新时间戳}// 获取编码后的输出数据var outputBufferIndex = encoder.dequeueOutputBuffer(bufferInfo, TIMEOUT_US)while (outputBufferIndex >= 0) {val outputBuffer = outputBuffers[outputBufferIndex]// 处理编码后的输出数据// 释放输出缓冲区encoder.releaseOutputBuffer(outputBufferIndex, false)outputBufferIndex = encoder.dequeueOutputBuffer(bufferInfo, TIMEOUT_US)}
}// 停止编码器并释放资源
encoder.stop()
encoder.release()
解码编码后的数据:
// 创建 MediaCodec 解码器,并配置解码器格式和参数
val decoder = MediaCodec.createDecoderByType("video/avc")
val mediaFormat = MediaFormat.createVideoFormat("video/avc", width, height)
mediaFormat.setByteBuffer("csd-0", csdBuffer) // 设置解码器参数
decoder.configure(mediaFormat, surface, null, 0) // 设置渲染 Surface
decoder.start()// 解码循环
while (isDecoding) {// 获取空闲的输入缓冲区索引val inputBufferIndex = decoder.dequeueInputBuffer(-1)if (inputBufferIndex >= 0) {val inputBuffer = inputBuffers[inputBufferIndex]// 将解码后的数据放入输入缓冲区decoder.queueInputBuffer(inputBufferIndex, 0, data.size, timestamp, 0)timestamp += 1_000_000 / frameRate // 更新时间戳}// 获取解码后的输出数据var outputBufferIndex = decoder.dequeueOutputBuffer(bufferInfo, TIMEOUT_US)while (outputBufferIndex >= 0) {val outputBuffer = outputBuffers[outputBufferIndex]// 处理解码后的输出数据// 渲染解码后的数据到 Surfacedecoder.releaseOutputBuffer(outputBufferIndex, true)outputBufferIndex = decoder.dequeueOutputBuffer(bufferInfo, TIMEOUT_US)}
}// 停止解码器并释放资源
decoder.stop()
decoder.release()
上述代码中的变量和参数需要根据你的实际情况进行调整。此外,NV21 格式的数据需要根据具体需要进行分割和处理传入编码器和解码器。