java 音频解码_java视频音频解码-封装xuggle-实现多种视频编码格式解码扩展

手头做一个视频相关项目,但是客户发来的测试视频(avi格式) 现有组件不能解码。现有

视频解码组件方案有基于JMF和opencv Jni调用。远远不能满足目前市面上玲琅满目的各种视频编码

标准。

进行检索 找到xuggler官方主页:http://www.xuggle.com/xuggler  对5.4版本进行简单封装,实现现有组件接口。需要slf4j包支持。实现了从现有组件只能支持摄像头和特定编码AVI文件到多种编码格式视频解码的支持。

经过测试至少支持 flv mov avi mpg wmv mp4 mkv 这些格式的视频解码。

实现代码如下:

package edu.zjut.framecollector;

import java.awt.image.BufferedImage;

import java.io.File;

import javax.media.ControllerEvent;

import javax.media.ControllerListener;

import javax.swing.JFileChooser;

import com.xuggle.xuggler.Global;

import com.xuggle.xuggler.ICodec;

import com.xuggle.xuggler.IContainer;

import com.xuggle.xuggler.IPacket;

import com.xuggle.xuggler.IPixelFormat;

import com.xuggle.xuggler.IStream;

import com.xuggle.xuggler.IStreamCoder;

import com.xuggle.xuggler.IVideoPicture;

import com.xuggle.xuggler.IVideoResampler;

import com.xuggle.xuggler.Utils;

/**

* @author 田旭园 E-mail: tianxuyuan@yahoo.com.cn

* @version 1.0 创建时间:2012-8-14 下午07:11:02

* @说明 AVI视频采集 解码 可对多种格式音频视频进行解码编码 http://blog.xuggle.com/

* @说明 JMF和opencv对各种视频编码格式的 不给力 学习AVIFrameCollector.java,进行对xuggler简单封装使用

* @说明 不需要有安装JMF 但是需要导入xuggle-xuggler-5.4.jar 和 slf4j库

*/

public class AVIFrameCollectorOnXuggle extends FrameCollector implements

ControllerListener {

private int step = 1;

IContainer container = null;

IStreamCoder videoCoder = null;

IPacket packet;

long firstTimestampInStream = Global.NO_PTS;

long systemClockStartTime = 0;

int videoStreamId = -1;

String filename;

IVideoResampler resampler = null;

@Override

public void close() {

if (videoCoder != null) {

videoCoder.close();

videoCoder = null;

}

if (container != null) {

container.close();

container = null;

}

}

@Override

public BufferedImage getCurrentFrame() {

BufferedImage javaImage = null;

while (container.readNextPacket(packet) >= 0) {

/*

* Now we have a packet, let's see if it belongs to our video stream

*/

if (packet.getStreamIndex() == videoStreamId) {

/*

* We allocate a new picture to get the data out of Xuggler

*/

IVideoPicture picture = IVideoPicture.make(videoCoder

.getPixelType(), videoCoder.getWidth(), videoCoder

.getHeight());

int offset = 0;

while (offset < packet.getSize()) {

/*

* Now, we decode the video, checking for any errors.

*/

int bytesDecoded = videoCoder.decodeVideo(picture, packet,

offset);

if (bytesDecoded < 0)

throw new RuntimeException(

"got error decoding video in: " + filename);

offset += bytesDecoded;

/*

* Some decoders will consume data in a packet, but will not

* be able to construct a full video picture yet. Therefore

* you should always check if you got a complete picture

* from the decoder

*/

if (picture.isComplete()) {

IVideoPicture newPic = picture;

/*

* If the resampler is not null, that means we didn't

* get the video in BGR24 format and need to convert it

* into BGR24 format.

*/

if (resampler != null) {

// we must resample

newPic = IVideoPicture.make(resampler

.getOutputPixelFormat(),

picture.getWidth(), picture.getHeight());

if (resampler.resample(newPic, picture) < 0)

throw new RuntimeException(

"could not resample video from: "

+ filename);

}

if (newPic.getPixelType() != IPixelFormat.Type.BGR24)

throw new RuntimeException("could not decode video"

+ " as BGR 24 bit data in: " + filename);

/**

* We could just display the images as quickly as we

* decode them, but it turns out we can decode a lot

* faster than you think.

*

* So instead, the following code does a poor-man's

* version of trying to match up the frame-rate

* requested for each IVideoPicture with the system

* clock time on your computer.

*

* Remember that all Xuggler IAudioSamples and

* IVideoPicture objects always give timestamps in

* Microseconds, relative to the first decoded item. If

* instead you used the packet timestamps, they can be

* in different units depending on your IContainer, and

* IStream and things can get hairy quickly.

*/

if (firstTimestampInStream == Global.NO_PTS) {

// This is our first time through

firstTimestampInStream = picture.getTimeStamp();

// get the starting clock time so we can hold up

// frames

// until the right time.

systemClockStartTime = System.currentTimeMillis();

} else {

long systemClockCurrentTime = System

.currentTimeMillis();

long millisecondsClockTimeSinceStartofVideo = systemClockCurrentTime

- systemClockStartTime;

// compute how long for this frame since the first

// frame in the

// stream.

// remember that IVideoPicture and IAudioSamples

// timestamps are

// always in MICROSECONDS,

// so we divide by 1000 to get milliseconds.

long millisecondsStreamTimeSinceStartOfVideo = (picture

.getTimeStamp() - firstTimestampInStream) / 1000;

final long millisecondsTolerance = 50; // and we

// give

// ourselfs

// 50 ms of

// tolerance

final long millisecondsToSleep = (millisecondsStreamTimeSinceStartOfVideo - (millisecondsClockTimeSinceStartofVideo + millisecondsTolerance));

if (millisecondsToSleep > 0) {

try {

Thread.sleep(millisecondsToSleep);

} catch (InterruptedException e) {

// we might get this when the user closes

// the dialog box, so

// just return from the method.

return null;

}

}

}

// And finally, convert the BGR24 to an Java buffered

// image

javaImage = Utils.videoPictureToImage(newPic);

return javaImage;

}

}

} else {

/*

* This packet isn't part of our video stream, so we just

* silently drop it.

*/

do {

} while (false);

}

}

return javaImage;

}

@Override

public FrameCollectorMode getMode() {

return FrameCollectorMode.AVI_FILE;

}

/**

*

* 使用用户定义的连接参数打开AVI视频文件。

*

*

* @param fileURL

* AVI视频文件的地址

* @param strStep

* 指定avi文件的播放速度,即每次跳帧的步进,调用时请注意给定的字符串要可以转换为整型。 1为正常速度,2为两倍速度,....

* @return true,如果成功打开,否则返回false

* @since 1.0

*/

@Override

public boolean open(String fileURL, String strStep) {

// fileURL="file:/F:/组件和项目/图像质量诊断工程/vedio/视频文件/亮度1.avi";

// 打开AVI视频文件

if (fileURL == null) {

return open();

}

if ((fileURL.substring(0, 6)).equals("file:/")) {

fileURL = fileURL.substring(6);

// System.out.println(" dfsfjasjf "+fileURL);

}

try {

step = Integer.parseInt(strStep);

} catch (NumberFormatException ex) {

step = 1;

}

return setupPlayer(fileURL);

}

/**

*

* 通过打开对话框打开指定AVI视频文件。

*

*

* @return true,如果成功打开,否则返回false

* @since 1.0

*/

@Override

public boolean open() {

// 从文件对话框中选择AVI文件

JFileChooser chooser = new JFileChooser();

chooser.setAcceptAllFileFilterUsed(false);

int result = chooser.showOpenDialog(null);

if (result == JFileChooser.CANCEL_OPTION) {

return false;

}

File file = chooser.getSelectedFile();

String fileURL = null;

try {

fileURL = file.getAbsolutePath();

System.out.println(fileURL);

} catch (Exception e) {

return false;

}

// 打开AVI视频文件

return setupPlayer(fileURL);

}

@Override

public void controllerUpdate(ControllerEvent arg0) {

// TODO Auto-generated method stub

}

/**

*

* Initialize the Player object.

*

*

* @param fileURL

* The selected file's URL

* @return true if set up the player successfully, false otherwise

*/

private boolean setupPlayer(String filename) {

this.filename = filename;

System.out.println(filename + " ==============");

// Let's make sure that we can actually convert video pixel formats.

if (!IVideoResampler

.isSupported(IVideoResampler.Feature.FEATURE_COLORSPACECONVERSION))

throw new RuntimeException("you must install the GPL version"

+ " of Xuggler (with IVideoResampler support) for "

+ "this demo to work");

// Create a Xuggler container object

container = IContainer.make();

// Open up the container

if (container.open(filename, IContainer.Type.READ, null) < 0)

throw new IllegalArgumentException("could not open file: "

+ filename);

// query how many streams the call to open found

int numStreams = container.getNumStreams();

// and iterate through the streams to find the first video stream

for (int i = 0; i < numStreams; i++) {

// Find the stream object

IStream stream = container.getStream(i);

// Get the pre-configured decoder that can decode this stream;

IStreamCoder coder = stream.getStreamCoder();

if (coder.getCodecType() == ICodec.Type.CODEC_TYPE_VIDEO) {

videoStreamId = i;

videoCoder = coder;

break;

}

}

if (videoStreamId == -1)

throw new RuntimeException(

"could not find video stream in container: " + filename);

/*

* Now we have found the video stream in this file. Let's open up our

* decoder so it can do work.

*/

if (videoCoder.open() < 0)

throw new RuntimeException(

"could not open video decoder for container: " + filename);

if (videoCoder.getPixelType() != IPixelFormat.Type.BGR24) {

// if this stream is not in BGR24, we're going to need to

// convert it. The VideoResampler does that for us.

resampler = IVideoResampler.make(videoCoder.getWidth(), videoCoder

.getHeight(), IPixelFormat.Type.BGR24, videoCoder

.getWidth(), videoCoder.getHeight(), videoCoder

.getPixelType());

if (resampler == null)

throw new RuntimeException("could not create color space "

+ "resampler for: " + filename);

}

/*

* Now, we start walking through the container looking at each packet.

*/

packet = IPacket.make();

return true;

}

}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/527623.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java 二叉树特点_二叉树的Java实现及特点总结

二叉树是一种非常重要的数据结构&#xff0c;它同时具有数组和链表各自的特点&#xff1a;它可以像数组一样快速查找&#xff0c;也可以像链表一样快速添加。但是他也有自己的缺点&#xff1a;删除操作复杂。我们先介绍一些关于二叉树的概念名词。二叉树&#xff1a;是每个结点…

java中session的作用_java中session的工作原理是什么?和Cookies有何区别?

现在大家学习的东西和以前大不相同了&#xff0c;越来越多的人倾向于去学习高新技术以获得更好的发展。java是很多人的第一选择。java中的知识还是很多的&#xff0c;今天就来为大家介绍一下。首先来说一下java中session的工作原理是什么?session的工作原理是客户端登录完成之…

java 大数四则运算_大数四则运算java(转)

1 //大数的四则运算2 #include 3 #include 4 #include 5 using namespace std;67 classBIGINTEGEROPERATIONS8 {9 private:10 static intCOMPARE(string number1, string number2)11 {12 intj;1314 int length1 number1.size();15 int length2 number2.size();1617 if(number1.…

http参数自动转换java接口参数设置_Springmvc请求参数类型转换器及原生api代码实例...

一、springmvc的xml配置文件xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xmlns:context"http://www.springframework.org/schema/context"xmlns:mvc"http://www.springframework.org/schema/mvc"xsi:schemaLocation"http://www.s…

java构建json_Java构造和解析Json数据的两种方法详解一

在www.json.org上公布了很多JAVA下的json构造和解析工具&#xff0c;其中org.json和json-lib比较简单&#xff0c;两者使用上差不多但还是有些区别。下面首先介绍用json-lib构造和解析Json数据的方法示例。用org.son构造和解析Json数据的方法详解请参见我下一篇博文&#xff1a…

python perl lisp,是否可能像python中的perl的lvalue或lisp的setf一样?

In lisp you can say:(setf (aref a 1) 5)In perl you can say:substr( $string, $start, $stop ) ~ s/a/b/gIs it possible something like this in python? I mean is it possible to use function result as a lvalue (as a target for assignment operation)?解决方案No.…

java final被覆盖_java中的final的使用

1、final类不能被继承&#xff0c;因此final类的成员方法没有机会被覆盖&#xff0c;默认都是final的。在设计类时候&#xff0c;如果这个类不需要有子类&#xff0c;类的实现细节不允许改变&#xff0c;并且确信这个类不会再被扩展&#xff0c;那么就设计为final类。(什么时候…

java jaxb注解xmlnull_java – 将空值表示为xml jaxb中的空元素

我强烈建议使用不存在节点或xsi&#xff1a;nil “true”属性来表示null.这最适用于模式验证(即< age />或< age>< / age>不是xsd&#xff1a;int类型的有效元素.但是,如果您不能在这里完成您的用例&#xff1a;标准JAXB行为import javax.xml.bind.annotatio…

wordcount.java_mapreduce中wordcount的java实现

用java模拟词频统计。有3个文件&#xff1a;text1: hello worldtext2:hello hadooptext3:hello mapreduce对上面的文件进行词频统计&#xff1a;结果应该是&#xff1a;hello:3; hadoop:1; world:1; mapreduce:1代码实现如下&#xff1a;package count;import java.ut…

java applog_java - 通过Logback登录到App Engine request_log - SO中文参考 - www.soinside.com

我当前部署的当前Logback配置如下(您可以说我的构想已耗尽...)&#xff1a;%-4relative [%thread] %-5level %logger{35} - %msgapplication.logWARNtech.provingground.dive_overwatch.etc.logging.enhancers.TestEnhancer%-4relative [%thread] %-5level %logger{35} - %msg查…

java生成pdf怎么合并行或者列_Java基础之PDF文件的合并

1、首先下载一个jar包&#xff1a;pdfbox-app-1.7.1.jar2、代码如下&#xff1a;package com;import java.io.File;import java.io.IOException;import org.apache.pdfbox.util.PDFMergerUtility;/*** PDF格式的图片合并**/public class PdfBox {private static String[] getFi…

java 迭代器的原理_java里Iterator的原理

Iterator&#xff1a;迭代器。其实就是集合取出元素的方式&#xff0c;每个容器的数据结构不一样&#xff0c;所以他们存取的方式不一样&#xff0c;具体事项方法不一样&#xff0c;具体实现方法不一样&#xff0c;每个集合都具备取出方式&#xff0c;对于取出这个动作不足以用…

Java里a和b哪个大_Java中 a+=b和a=a+b有什么区别?

在java语言中&#xff0c;ab和aab的主要区别是在运算的精度上。类似的有“- 、 * 、/ 、% ”&#xff0c;这里以ab 与 aab 举例说明一下。1.下面是一串代码&#xff0c;我们试一下aab的形式&#xff1a;public class TestDemo {public static void main(String[] args) {byte a…

java程序回滚之后在哪看_Java在触发事务回滚之后为什么会再一次回到Servlet开始的地方重新走一次流程?...

代码流程前台点击"提交订单"进入BaseServlet.classBaseServlet.class分发至子类OrderServlet.class的submitOrder()方法submitOrder()调用Service层的submitOrder()方法.关键是Service层submitOrder()中使用了事务回滚. 这里调用了Dao层两个方法: fun01()和fun02(), …

java不进入for_为什么阿里巴巴Java开发手册中强制要求不要在foreach循环里进行元素的remove和add操作?...

在阅读《阿里巴巴Java开发手册》时&#xff0c;发现有一条关于在 foreach 循环里进行元素的 remove/add 操作的规约&#xff0c;具体内容如下&#xff1a;错误演示我们首先在 IDEA 中编写一个在 foreach 循环里进行 remove 操作的代码&#xff1a;import java.util.ArrayList;i…

mysql对测试如何_我如何对MySQL进行基准测试?

我目前正在使用MySQL工作台.我希望看到表中行数增加时性能的差异.我想专门测试并比较1000行,10,000行,100,000行,1,000,000行和10,000,000行.那么,是否有任何工具可以让我这样做并提供有关磁盘I / O,内存使用情况,CPU使用率和完成查询的时间的统计信息&#xff1f;解决方法:是.…

JAVA两个视图层_MVC - 管理帐户 . 一个视图有两个局部视图和两个模型

美好的一天&#xff01;我正在管理帐户页面上&#xff0c;用户可以在其中更改密码和电子邮件 .我有一个管理视图&#xff0c;其中有两个部分视图&#xff0c;更改密码和更改电子邮件 . 这两种都使用两种不同的模型 . 这样做的目的是在同一页面上同时包含更改密码和更改电子邮件…

8086汇编4位bcd码_二进制格雷码与自然二进制码的互换分析

在精确定位控制系统中&#xff0c;为了提高控制精度&#xff0c;准确测量控制对象的位置是十分重要的。目前&#xff0c;检测位置的办法有两种&#xff1a;其一是使用位置传感器&#xff0c;测量到的位移量由变送器经A/D转换成数字量送至系统进行进一步处理。此方法精度高&…

python datetime.datetime 当前_关于datetime:如何让python显示当前时间(东部)

如何让python在东方显示时间&#xff1f;我已经看过了Python文档&#xff0c;但它非常混乱。我使用的是python 3。谢谢。这可能有帮助&#xff1a;stackoverflow.com/questions/117514/&hellip&#xff1b;基本上&#xff0c;在内部使用UTC&#xff0c;并在显示时使用pytz转…

软件工程结构化建模的方法和工具_软件工程系列-结构化设计方法2

本系列文章为笔记&#xff0c;内容根据北京大学《软件工程》MOOC 初始化模块结构图精化的启发式规则常见的启发式规则什么叫做“启发式”根据设计准则&#xff0c;从长期的软件开发实践中&#xff0c;总结出来的规则既不是设计目标&#xff0c;也不是设计时应该普遍遵循的原理常…