当前位置:首页 » 安卓系统 » androidmediacodec解码

androidmediacodec解码

发布时间: 2022-09-23 00:33:22

A. android使用mediacodec进行编解码 常用吗

应该还是比较常用的。示例如下:

java">privatefinalStringTAG="MediaCodeSample";
/**用来解码*/
privateMediaCodecmMediaCodec;
/**用来读取音频文件*/
;
privateMediaFormatformat;
privateStringmime=null;
privateintsampleRate=0,channels=0,bitrate=0;
privatelongpresentationTimeUs=0,ration=0;
publicvoiddecode(Stringurl)
{

extractor=newMediaExtractor();
//根据路径获取源文件
try
{
extractor.setDataSource(url);
}catch(Exceptione)
{
Log.e(TAG,"设置文件路径错误"+e.getMessage());
}
try
{
//音频文件信息
format=extractor.getTrackFormat(0);
mime=format.getString(MediaFormat.KEY_MIME);
sampleRate=format.getInteger(MediaFormat.KEY_SAMPLE_RATE);
//声道个数:单声道或双声道
channels=format.getInteger(MediaFormat.KEY_CHANNEL_COUNT);
//ifrationis0,
ration=format.getLong(MediaFormat.KEY_DURATION);
//System.out.println("歌曲总时间秒:"+ration/1000000);
bitrate=format.getInteger(MediaFormat.KEY_BIT_RATE);
}catch(Exceptione)
{
Log.e(TAG,"音频文件信息读取出错:"+e.getMessage());
//不要退出,下面进行判断
}
Log.d(TAG,"Trackinfo:mime:"+mime+"采样率sampleRate:"+sampleRate+"channels:"+channels+"bitrate:"
+bitrate+"ration:"+ration);
//检查是否为音频文件
if(format==null||!mime.startsWith("audio/"))
{
Log.e(TAG,"不是音频文件end!");
return;
}
//实例化一个指定类型的解码器,提供数据输出
//
mMediaCodec=MediaCodec.createDecoderByType(mime);

if(mMediaCodec==null)
{
Log.e(TAG,"创建解码器失败!");
return;
}
mMediaCodec.configure(format,null,null,0);

mMediaCodec.start();
//用来存放目标文件的数据
ByteBuffer[]inputBuffers=mMediaCodec.getInputBuffers();
//解码后的数据
ByteBuffer[]outputBuffers=mMediaCodec.getOutputBuffers();
//设置声道类型:AudioFormat.CHANNEL_OUT_MONO单声道,AudioFormat.CHANNEL_OUT_STEREO双声道
intchannelConfiguration=channels==1?AudioFormat.CHANNEL_OUT_MONO:AudioFormat.CHANNEL_OUT_STEREO;
Log.i(TAG,"channelConfiguration="+channelConfiguration);
extractor.selectTrack(0);
//==========开始解码=============
booleansawInputEOS=false;
booleansawOutputEOS=false;
finallongkTimeOutUs=10;
MediaCodec.BufferInfoinfo=newMediaCodec.BufferInfo();
while(!sawOutputEOS)
{
try
{
if(!sawInputEOS)
{
intinputBufIndex=mMediaCodec.dequeueInputBuffer(kTimeOutUs);
if(inputBufIndex>=0)
{
ByteBufferdstBuf=inputBuffers[inputBufIndex];

intsampleSize=extractor.readSampleData(dstBuf,0);
if(sampleSize<0)
{
Log.d(TAG,"sawinputEOS.Stoppingplayback");
sawInputEOS=true;
sampleSize=0;
}else
{
presentationTimeUs=extractor.getSampleTime();
}

mMediaCodec.queueInputBuffer(inputBufIndex,0,sampleSize,presentationTimeUs,
sawInputEOS?MediaCodec.BUFFER_FLAG_END_OF_STREAM:0);

if(!sawInputEOS)
{
extractor.advance();
}

}else
{
Log.e(TAG,"inputBufIndex"+inputBufIndex);
}
}//!sawInputEOS

//
intres=mMediaCodec.dequeueOutputBuffer(info,kTimeOutUs);

if(res>=0)
{
intoutputBufIndex=res;
ByteBufferbuf=outputBuffers[outputBufIndex];
finalbyte[]chunk=newbyte[info.size];
buf.get(chunk);
buf.clear();
if(chunk.length>0)
{

//chunk解码后的音频流
//TODO:处理...
}
mMediaCodec.releaseOutputBuffer(outputBufIndex,false);
if((info.flags&MediaCodec.BUFFER_FLAG_END_OF_STREAM)!=0)
{
Log.d(TAG,"sawoutputEOS.");
sawOutputEOS=true;
}

}elseif(res==MediaCodec.INFO_OUTPUT_BUFFERS_CHANGED)
{
outputBuffers=mMediaCodec.getOutputBuffers();
Log.w(TAG,"[AudioDecoder]outputbuffershavechanged.");
}elseif(res==MediaCodec.INFO_OUTPUT_FORMAT_CHANGED)
{
MediaFormatoformat=mMediaCodec.getOutputFormat();
Log.w(TAG,"[AudioDecoder]outputformathaschangedto"+oformat);
}else
{
Log.w(TAG,"[AudioDecoder]dequeueOutputBufferreturned"+res);
}

}catch(RuntimeExceptione)
{
Log.e(TAG,"[decodeMP3]error:"+e.getMessage());
}
}
//=================================================================================
if(mMediaCodec!=null)
{
mMediaCodec.stop();
mMediaCodec.release();
mMediaCodec=null;
}
if(extractor!=null)
{
extractor.release();
extractor=null;
}
//clearsourceandtheotherglobals
ration=0;
mime=null;
sampleRate=0;
channels=0;
bitrate=0;
presentationTimeUs=0;
ration=0;
}

B. android使用mediacodec进行编解码 常用吗

应该还是比较常用的。示例如下:
private final String TAG = "MediaCodeSample";
/** 用来解码 */
private MediaCodec mMediaCodec;
/** 用来读取音频文件 */
private MediaExtractor extractor;
private MediaFormat format;
private String mime = null;
private int sampleRate = 0, channels = 0, bitrate = 0;
private long presentationTimeUs = 0, ration = 0;
public void decode(String url)
{

extractor = new MediaExtractor();
// 根据路径获取源文件
try
{
extractor.setDataSource(url);
} catch (Exception e)
{
Log.e(TAG, " 设置文件路径错误" + e.getMessage());
}
try
{
// 音频文件信息
format = extractor.getTrackFormat(0);
mime = format.getString(MediaFormat.KEY_MIME);
sampleRate = format.getInteger(MediaFormat.KEY_SAMPLE_RATE);
// 声道个数:单声道或双声道
channels = format.getInteger(MediaFormat.KEY_CHANNEL_COUNT);
// if ration is 0, we are probably playing a live stream
ration = format.getLong(MediaFormat.KEY_DURATION);
// System.out.println("歌曲总时间秒:"+ration/1000000);
bitrate = format.getInteger(MediaFormat.KEY_BIT_RATE);
} catch (Exception e)
{
Log.e(TAG, "音频文件信息读取出错:" + e.getMessage());
// 不要退出,下面进行判断
}
Log.d(TAG, "Track info: mime:" + mime + " 采样率sampleRate:" + sampleRate + " channels:" + channels + " bitrate:"
+ bitrate + " ration:" + ration);
// 检查是否为音频文件
if (format == null || !mime.startsWith("audio/"))
{
Log.e(TAG, "不是音频文件 end !");
return;
}
// 实例化一个指定类型的解码器,提供数据输出
// Instantiate an encoder supporting output data of the given mime type
mMediaCodec = MediaCodec.createDecoderByType(mime);

if (mMediaCodec == null)
{
Log.e(TAG, "创建解码器失败!");
return;
}
mMediaCodec.configure(format, null, null, 0);

mMediaCodec.start();
// 用来存放目标文件的数据
ByteBuffer[] inputBuffers = mMediaCodec.getInputBuffers();
// 解码后的数据
ByteBuffer[] outputBuffers = mMediaCodec.getOutputBuffers();
// 设置声道类型:AudioFormat.CHANNEL_OUT_MONO单声道,AudioFormat.CHANNEL_OUT_STEREO双声道
int channelConfiguration = channels == 1 ? AudioFormat.CHANNEL_OUT_MONO : AudioFormat.CHANNEL_OUT_STEREO;
Log.i(TAG, "channelConfiguration=" + channelConfiguration);
extractor.selectTrack(0);
// ==========开始解码=============
boolean sawInputEOS = false;
boolean sawOutputEOS = false;
final long kTimeOutUs = 10;
MediaCodec.BufferInfo info = new MediaCodec.BufferInfo();
while (!sawOutputEOS)
{
try
{
if (!sawInputEOS)
{
int inputBufIndex = mMediaCodec.dequeueInputBuffer(kTimeOutUs);
if (inputBufIndex >= 0)
{
ByteBuffer dstBuf = inputBuffers[inputBufIndex];

int sampleSize = extractor.readSampleData(dstBuf, 0);
if (sampleSize < 0)
{
Log.d(TAG, "saw input EOS. Stopping playback");
sawInputEOS = true;
sampleSize = 0;
} else
{
presentationTimeUs = extractor.getSampleTime();
}

mMediaCodec.queueInputBuffer(inputBufIndex, 0, sampleSize, presentationTimeUs,
sawInputEOS ? MediaCodec.BUFFER_FLAG_END_OF_STREAM : 0);

if (!sawInputEOS)
{
extractor.advance();
}

} else
{
Log.e(TAG, "inputBufIndex " + inputBufIndex);
}
} // !sawInputEOS

// decode to PCM and push it to the AudioTrack player
int res = mMediaCodec.dequeueOutputBuffer(info, kTimeOutUs);

if (res >= 0)
{
int outputBufIndex = res;
ByteBuffer buf = outputBuffers[outputBufIndex];
final byte[] chunk = new byte[info.size];
buf.get(chunk);
buf.clear();
if (chunk.length > 0)
{

// chunk解码后的音频流
// TODO:处理...
}
mMediaCodec.releaseOutputBuffer(outputBufIndex, false);
if ((info.flags & MediaCodec.BUFFER_FLAG_END_OF_STREAM) != 0)
{
Log.d(TAG, "saw output EOS.");
sawOutputEOS = true;
}

} else if (res == MediaCodec.INFO_OUTPUT_BUFFERS_CHANGED)
{
outputBuffers = mMediaCodec.getOutputBuffers();
Log.w(TAG, "[AudioDecoder]output buffers have changed.");
} else if (res == MediaCodec.INFO_OUTPUT_FORMAT_CHANGED)
{
MediaFormat oformat = mMediaCodec.getOutputFormat();
Log.w(TAG, "[AudioDecoder]output format has changed to " + oformat);
} else
{
Log.w(TAG, "[AudioDecoder] dequeueOutputBuffer returned " + res);
}

} catch (RuntimeException e)
{
Log.e(TAG, "[decodeMP3] error:" + e.getMessage());
}
}
// =================================================================================
if (mMediaCodec != null)
{
mMediaCodec.stop();
mMediaCodec.release();
mMediaCodec = null;
}
if (extractor != null)
{
extractor.release();
extractor = null;
}
// clear source and the other globals
ration = 0;
mime = null;
sampleRate = 0;
channels = 0;
bitrate = 0;
presentationTimeUs = 0;
ration = 0;
}

C. Android MediaCodec

MediaCodec 类为开发者提供了能访问到Android底层媒体 Codec (Encoder/Decoder)的能力,它是Android底层多媒体基础架构的一部分(通常和MediaExtractor、MediaSync、MediaMuxer、MediaCrypto、MediaDrm、Image、Surface、AudioTrack一起使用)。

Codec 对三种类型类型的数据起作用: 编码后的压缩数据 , 原始视频数据 , 原始音频数据 。这三种类型的数据都可以通过 ByteBuffer 来传递给 Codec ,但是对于 原始视频数据 我们建议使用 Surface 来传递,这样可以提高 Codec 的性能, Surface 使用的是 native video buffer ,不用映射或者拷贝成 ByteBuffer ,因此这样的方式更高效。当你使用 Surface 来传递 原始视频数据 时,也就无法获取到了 原始视频数据 ,Android 提供了 ImageReader 帮助你获取到解码后的 原始视频数据 。这种方式可能仍然有要比 ByteBuffer 的方式更加高效,因为某些 native video buffer 会直接映射成 byteBuffer 。当然如果你 ByteBuffer 的模式,你可以使用 Image 类提供的 getInput/OutputImage(int) 来获取 原始视频数据 。

给 Decoder 输入的 InputBuffer 或者 Encoder 输出的 outputBuffer 包含的都是编码后的压缩数据,数据的压缩类型由 MediaFormat#KEY_MIME 指明。对于视频类型而言,这个数据通常是一个压缩后的视频帧。对于音频数据而言,通常是一个访问单元(一个编码的音频段,通常包含几毫秒的音频数据,数据类型format type 指定),有时候,一个音频单元对于一个 buffer 而言可能有点宽松,所以一个 buffer 里可能包含多个编码后的音频数据单元。无论 Buffer 包含的是视频数据还是音频数据, Buffer 都不会再任意字节边界上开始或者结束,而是在帧(视频)或者单元(音频)的边界上开始或者结束。除非它们被BUFFER_FLAG_PARTIAL_FRAME标记。

原始音频Buffer包含PCM音频数据的整个帧,是每一个通道按着通道顺序的采样数据。每一个采样按16Bit量化。

在 ByteBuffer 模式下,视频数据的排布由 MediaFormat#KEY_COLOR_FORMAT 指定,我们可以通过 getCodecInfo().MediaCodecInfo#getCapabilitiesForType.CodecCapabilities#colorFormats 获取到一个设备支持的 color format 数组。视频 Codec 可能支持三种类型的Color Format:

从 Build.VERSION_CODES.LOLLIPOP_MR1 开始所有的视频 Codec 都支持 flexible YUV 4:2:0

对于 Build.VERSION_CODES.LOLLIPOP 之前并且支持 Image 类时,我们需要使用 MediaFormat#KEY_STRIDE 和 MediaFormat#KEY_SLICE_HEIGHT 的值去理解输出的原始视频数据的布局。

键值 MediaFormat#KEY_WIDTH 和 MediaFormat#KEY_HEIGHT 指明了视频Frame的size。然而,对于大多数用于编码的视频图像,他们只占用了video Frame的一部分。这部分用一个 'crop rectangle 来表示。

我们需要用下面的一些 keys 从获取原始视频数据的 crop rectangle ,如果 out format 中没有包含这些 keys ,则表示视频占据了整个 video Frame ,这个 crop rectangle 的解释应该立足于应用任何 MediaFormat#KEY_ROTATION 之前。

下面是在旋转之前计算视频的尺寸的案例:

从概念上讲Codec的声明周期存在三种状态: Stoped , Executing , Released 。 Stoped 状态是一个集合状态,它聚合了三种状态: Uninitialized , Configured ,和 Error ,同时 Executing 状态的处理也是通过三个子状态来完成: Flushed , Running , End-of-Stream 。

Executing 状态有三个子状态:Flushed,Running,和End-of-Stream,当我们调用玩 Start() 函数后, Codec 就立刻进入 Flushed 子状态,这个状态下,它持有全部的buffer,只要第一个Input buffer被dequeued,Codec就转变成 Running 子状态,这个状态占据了 Codec 的生命周期的绝大部分。当入队一个带有 end-of-stream标志的InputBuffer后, Codec 将转换成 End of Stream 子状态,在这个状态下, Codec 将不会再接收任何输入的数据,但是仍然会产生output buffer ,直到end-of-Stream标记的buffer被输出。我们可以在 Executing 状态的任何时候,使用 flush() 函数,将 Codec 切换成 Flushed 状态。

调用 stop() 函数会将 Codec 返回到 Uninitialized 状态,这样我们就可以对 Codec 进行重新配置,当你用完了 Codec 后,你必须要调用 release() 函数去释放这个 Codec 。

在极少数情况下, Codec 可能也会遇到错误,此时 Codec 将会切换到 Error 状态,我们可以通过queuing操作获取到一个无效的返回值,或者有时会通过异常来的得知 Codec 发生了错误。通过调用 reset() 函数,将 Codec 进行重置,这样 Codec 将切换成 Uninitalized 状态,我们可以在任何状态下调用 rest() 函数将Codec 将切换成 Uninitalized`状态。

使用 MediaCodecList 创建一个指定 MediaFormat 的MediaCodec。当我们解码一个文件或者一个流时,我们可以通过 MediaExtractor#getTrackFormat 获取期望的Fromat,同时我们可以通过 MediaFormat#setFeatureEnabled 为 Codec 注入任何我们想要的特性。然后调用 MediaCodecList#findDecoderForFormat 获取能够处理对应format数据 Codec 的name,最后我们使用 createByCodecName(String) 创建出这个 Codec 。

我们也可以使用 createDecoder/EncoderByType(java.lang.String) 函数来创建指定的 MIME 类型的 Codec ,但是这样我们无法向其中注入一些指定的特性,这样创建的 Codec 可能不能处理我们期望的媒体类型数据。

D. android使用MediaCodec进行解码,就是硬解码吗

Android api中有自带的硬解码方法MediaCodec,但是SDK并没有规定输出格式,参考H264的解码格式

E. 手机自带mediacodec吗

手机自带mediacodec。MediaCodec 是 Android 中的编解码器组件,用来访问底层提供的编解码器,手机是自带mediacodec的。

通常与MediaExtractor、MediaSync、MediaMuxer、MediaCrypto、MediaDrm、Image、Surface和AudioTrack一起使用,MediaCodec几乎是Android播放器硬解码的标配。

多媒体芯片的发展

在1996年MMX CPU问世之际,即打乱了多媒体芯片发展的时间表。被英特尔视为近10年来CPU革命性进展的大作MMX,借由CPU中新增的57个多媒体指令集、8个64位MMX暂存器及32 KB的Cache。

不仅提升了多媒体及通信程序的执行性能,亦颠覆了多媒体及通信产业的生态环境,且以后CPU亦将遵循此方向发展。

软件MPEG将取代硬件MPEGMPEG—l硬件解决方案在CPU功能较强的情况下,被软件MPEG取代掉了。MPEG2亦面临岌岌可危之境。MPEG硬件将谋求PC领域以外的新世界,转向进攻消费性电子产品。

低级Audio将遭受危机MMX CPU取代声卡。此时强调高品质、高层次的音效芯片将有脱颖而出的机会。未来可能将以On board或IC整合的形式,对声卡厂商造成不小的冲击,从而思索未来转型之道。

性能的提高MMX旨在增进绘图卡性能,而AGP也已成为提升系统芯片组与绘图芯片问的桥梁。因此,绘图卡将仍会持续,在绘图芯片上将朝影像、动画、图形加速、3D方向整合MMX,辅以影像绘图加速j枣片将使PC有更大的发挥。

热点内容
源码车牌识别 发布:2025-01-13 19:49:06 浏览:974
inflateandroid 发布:2025-01-13 19:47:43 浏览:244
安卓电脑如何连手机热点 发布:2025-01-13 19:40:52 浏览:142
pythonnumexpr 发布:2025-01-13 19:34:56 浏览:489
linuxpython版本查看 发布:2025-01-13 19:18:37 浏览:743
宝马三系的哪个配置走量 发布:2025-01-13 19:12:14 浏览:697
局域网如何访问服务器 发布:2025-01-13 18:56:24 浏览:191
javaarraylist访问 发布:2025-01-13 18:56:10 浏览:671
如何确定胶体是否配置成功 发布:2025-01-13 18:46:07 浏览:73
fgo缓存多大 发布:2025-01-13 18:42:13 浏览:974