Android c++中的NdkMediaCodec与MediaCodec有什么联系或区别
时间: 2024-04-02 07:34:26 浏览: 272
NdkMediaCodec是Android NDK提供的媒体编解码器接口,而MediaCodec是Android SDK提供的媒体编解码器接口。它们之间的联系是NdkMediaCodec是在MediaCodec之上封装的,具有更高效、更底层的特性,适用于需要更高性能的场景。区别在于NdkMediaCodec需要手动管理内存,需要更多的代码来实现功能,而MediaCodec则相对简单易用,但也有一定的性能限制。另外,NdkMediaCodec只能在C++中使用,而MediaCodec可以在Java中使用。
相关问题
Android 使用c++中的MediaCodec解码的完整代码,注意不要使用NdkMediaCodec
以下是使用C++中的MediaCodec解码的Android完整代码示例:
```c++
#include <jni.h>
#include <android/log.h>
#include <android/native_window_jni.h>
#include <media/NdkMediaCodec.h>
#include <media/NdkMediaExtractor.h>
#define LOG_TAG "MediaCodec"
#define LOGI(...) ((void)__android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__))
#define LOGW(...) ((void)__android_log_print(ANDROID_LOG_WARN, LOG_TAG, __VA_ARGS__))
extern "C" JNIEXPORT void JNICALL
Java_com_example_mediadecoder_MainActivity_playMedia(JNIEnv* env, jobject obj, jstring path, jobject surface)
{
const char* filePath = env->GetStringUTFChars(path, nullptr);
ANativeWindow* window = ANativeWindow_fromSurface(env, surface);
AMediaExtractor* extractor = AMediaExtractor_new();
media_status_t status = AMediaExtractor_setDataSource(extractor, filePath);
if (status != AMEDIA_OK) {
LOGW("Failed to set data source: %d", status);
return;
}
int32_t trackCount = AMediaExtractor_getTrackCount(extractor);
AMediaCodec* codec = nullptr;
AMediaFormat* format = nullptr;
ssize_t trackIndex = -1;
for (int32_t i = 0; i < trackCount; ++i) {
AMediaFormat* trackFormat = AMediaExtractor_getTrackFormat(extractor, i);
const char* mime = nullptr;
AMediaFormat_getString(trackFormat, AMEDIAFORMAT_KEY_MIME, &mime);
if (strncmp(mime, "video/", 6) == 0) {
trackIndex = i;
format = trackFormat;
break;
}
AMediaFormat_delete(trackFormat);
}
if (trackIndex != -1) {
AMediaExtractor_selectTrack(extractor, trackIndex);
codec = AMediaCodec_createDecoderByType(format);
AMediaCodec_configure(codec, format, window, nullptr, 0);
AMediaCodec_start(codec);
int64_t timeoutUs = 10000;
AMediaCodecBufferInfo bufferInfo = {0};
ssize_t sampleSize = 0;
while ((sampleSize = AMediaExtractor_readSampleData(extractor, AMediaCodec_getInputBuffer(codec, 0), 0)) >= 0) {
uint8_t* data = nullptr;
size_t size = 0;
AMediaCodecBufferInfo info = {0};
ssize_t index = AMediaCodec_dequeueInputBuffer(codec, timeoutUs);
if (index >= 0) {
auto inputBuffer = AMediaCodec_getInputBuffer(codec, index);
if (inputBuffer != nullptr) {
data = AMediaExtractor_getBuffer(extractor);
size = AMediaExtractor_getSampleSize(extractor);
info.offset = 0;
info.size = size;
info.presentationTimeUs = AMediaExtractor_getSampleTime(extractor);
info.flags = AMediaExtractor_getSampleFlags(extractor);
memcpy(data, inputBuffer, size);
AMediaCodec_queueInputBuffer(codec, index, 0, size, info.presentationTimeUs, info.flags);
AMediaExtractor_advance(extractor);
}
}
ssize_t outIndex = AMediaCodec_dequeueOutputBuffer(codec, &bufferInfo, timeoutUs);
if (outIndex >= 0) {
auto outputBuffer = AMediaCodec_getOutputBuffer(codec, outIndex);
if (outputBuffer != nullptr) {
AMediaCodec_releaseOutputBuffer(codec, outIndex, true);
}
}
}
AMediaCodec_stop(codec);
AMediaCodec_delete(codec);
}
AMediaExtractor_delete(extractor);
ANativeWindow_release(window);
env->ReleaseStringUTFChars(path, filePath);
}
```
这段代码通过使用Android NDK的MediaCodec和MediaExtractor类来读取视频文件并将其渲染到给定的Surface上。在代码中,先使用AMediaExtractor_setDataSource()函数来设置数据源,然后扫描所有的轨道,找到视频轨道并选择它。接下来,使用AMediaCodec_createDecoderByType()函数创建解码器,然后使用AMediaCodec_configure()函数配置解码器并启动它。之后,使用AMediaCodec_getInputBuffer()函数获取解码器的输入缓冲区,并使用AMediaExtractor_readSampleData()函数从媒体文件中读取样本数据并将其复制到解码器的输入缓冲区中。最后,使用AMediaCodec_dequeueOutputBuffer()函数获取解码器的输出缓冲区,并使用ANativeWindow_lock()函数将视频帧渲染到给定的Surface上。
注意:以上代码仅为示例代码,并未在实际项目中测试过,可能存在bug和不足之处。
Android中,使用C++类MediaCodec,调用MediaCodec::getOutputBuffer的解码例子
以下是一个简单的Android JNI代码示例,演示如何使用C++类MediaCodec和MediaFormat来解码视频文件并获取输出缓冲区:
```c++
#include <jni.h>
#include <android/log.h>
#include <android/native_window_jni.h>
#include <media/NdkMediaCodec.h>
#define LOG_TAG "MediaCodecExample"
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)
extern "C" {
JNIEXPORT void JNICALL
Java_com_example_MediaCodecUtils_decode(JNIEnv *env, jobject instance, jstring filePath_,
jobject surface) {
const char *filePath = env->GetStringUTFChars(filePath_, NULL);
FILE *fp = fopen(filePath, "rb");
if (!fp) {
LOGD("Failed to open file: %s", filePath);
return;
}
// 创建解码器
AMediaCodec *codec = AMediaCodec_createDecoderByType("video/avc");
AMediaFormat *format = AMediaFormat_new();
AMediaFormat_setString(format, AMEDIAFORMAT_KEY_MIME, "video/avc");
AMediaFormat_setInt32(format, AMEDIAFORMAT_KEY_WIDTH, 1920);
AMediaFormat_setInt32(format, AMEDIAFORMAT_KEY_HEIGHT, 1080);
AMediaFormat_setBuffer(format, "csd-0", (void *) csd_data, csd_size);
AMediaCodec_configure(codec, format, ANativeWindow_fromSurface(env, surface),
NULL /* crypto */, 0 /* flags */);
AMediaCodec_start(codec);
// 循环解码
bool inputDone = false;
bool outputDone = false;
while (!outputDone) {
if (!inputDone) {
// 获取可用输入缓冲区的索引
ssize_t inputIndex = AMediaCodec_dequeueInputBuffer(codec, 2000);
if (inputIndex >= 0) {
AMediaCodecBufferInfo inputBufferInfo;
auto inputBuffer = AMediaCodec_getInputBuffer(codec, inputIndex);
size_t bytesRead = fread(inputBuffer->data, 1, inputBuffer->capacity, fp);
if (bytesRead > 0) {
// 提交输入缓冲区
AMediaCodec_queueInputBuffer(codec, inputIndex, 0, bytesRead, 0,
inputDone ? AMEDIACODEC_BUFFER_FLAG_END_OF_STREAM : 0);
} else {
inputDone = true;
AMediaCodec_queueInputBuffer(codec, inputIndex, 0, 0, 0, AMEDIACODEC_BUFFER_FLAG_END_OF_STREAM);
}
}
}
// 获取可用输出缓冲区的索引
AMediaCodecBufferInfo outputBufferInfo;
ssize_t outputIndex = AMediaCodec_dequeueOutputBuffer(codec, &outputBufferInfo, 0);
if (outputIndex == AMEDIACODEC_INFO_OUTPUT_FORMAT_CHANGED) {
// 更新输出格式
format = AMediaCodec_getOutputFormat(codec);
LOGD("Output format changed: %s", AMediaFormat_toString(format));
} else if (outputIndex == AMEDIACODEC_INFO_OUTPUT_BUFFERS_CHANGED) {
// 忽略此消息
LOGD("Output buffers changed");
} else if (outputIndex >= 0) {
auto outputBuffer = AMediaCodec_getOutputBuffer(codec, outputIndex);
// 处理输出缓冲区
LOGD("Output buffer %zd, size %d, flags %d, pts %lld, dts %lld", outputIndex,
outputBufferInfo.size, outputBufferInfo.flags, outputBufferInfo.presentationTimeUs,
outputBufferInfo.decodingTimeUs);
// 获取输出缓冲区的数据
uint8_t *data = outputBuffer->data;
int dataSize = outputBufferInfo.size;
// 处理数据...
// 处理完输出缓冲区后,释放缓冲区
AMediaCodec_releaseOutputBuffer(codec, outputIndex, true /* render */);
outputDone = (outputBufferInfo.flags & AMEDIACODEC_BUFFER_FLAG_END_OF_STREAM) != 0;
}
}
// 释放资源
AMediaCodec_stop(codec);
AMediaCodec_delete(codec);
fclose(fp);
env->ReleaseStringUTFChars(filePath_, filePath);
}
}
```
这个例子中,我们使用AMediaCodec_createDecoderByType函数创建一个H.264视频解码器,然后使用AMediaFormat来指定视频的宽度、高度、MIME类型和CSD数据。在configure方法中,我们将解码器与一个Surface关联起来,以便输出图像可以在屏幕上渲染。在循环中,我们首先获取可用的输入缓冲区索引,并从输入文件中读取数据填充缓冲区。然后,我们提交输入缓冲区,并等待解码器输出缓冲区。当我们获取可用的输出缓冲区索引时,我们可以从输出缓冲区中获取数据并进行处理。最后,我们释放输出缓冲区并检查是否到达了输入文件的结尾。
阅读全文