Xuggler教程:转码和媒体修改

注意:这是我们的“ Xuggler开发教程 ”系列的一部分。

在之前的教程中,我对视频处理Xuggler进行了简短介绍 。 在这一部分中,我们将看到Xuggler和FFmpeg提供的一些更令人兴奋的功能,例如视频转码和媒体修改。 不要忘记Xuggler是一个Java库,可用于实时解压缩,处理和压缩录制的视频或实时视频。

Xuggler提供了两种不同的编程API,可用于同一目的。 首先,我们有MediaTool API :

MediaTool是一个简单的应用程序编程接口(API),用于对Java中的视频进行解码,编码和修改。 MediaTool隐藏了许多容器,编解码器和其他内容的细节,因此您可以专注于媒体而不是工具。 就是说,MediaTool仍然提供对基础Xuggler对象的访问,因此,如果需要,您可以进行精细的控制。

还有Xuggler Advanced API ,它使您可以深入研究视频操作的细节,但又增加了一层复杂性。

首先,我们将使用MediaTool API ,在随后的教程中,我们还将处理Advanced API。

让我们开始将媒体从一种格式转码为另一种格式。 代码转换是一种编码到另一种编码的直接数模转换。 通常在目标设备不支持格式或存储容量有限(要求减小文件大小)或将不兼容或过时的数据转换为更好支持的格式或现代格式的情况下执行此操作。 转码通常是一个有损过程 ,其中“有损”压缩是一种数据编码方法,为了达到其目标,它会丢弃(丢失)某些数据,结果是,对数据进行解压缩会产生与原始数据不同的内容,尽管足够相似,以某种方式有用。

让我们看一些用于转码的高级代码,稍后我将详细解释。

package com.javacodegeeks.xuggler;import com.xuggle.mediatool.IMediaReader;
import com.xuggle.mediatool.IMediaViewer;
import com.xuggle.mediatool.IMediaWriter;
import com.xuggle.mediatool.ToolFactory;public class TranscodingExample {private static final String inputFilename = "c:/myvideo.mp4";private static final String outputFilename = "c:/myvideo.flv";public static void main(String[] args) {// create a media readerIMediaReader mediaReader = ToolFactory.makeReader(inputFilename);// create a media writerIMediaWriter mediaWriter = ToolFactory.makeWriter(outputFilename, mediaReader);// add a writer to the reader, to create the output filemediaReader.addListener(mediaWriter);// create a media viewer with stats enabledIMediaViewer mediaViewer = ToolFactory.makeViewer(true);// add a viewer to the reader, to see the decoded mediamediaReader.addListener(mediaViewer);// read and decode packets from the source file and// and dispatch decoded audio and video to the writerwhile (mediaReader.readPacket() == null) ;}}

只需几行代码,我们就可以将MPEG-4输入文件转换为FLV文件。 我们首先创建一个IMediaReader ,用于读取和解码媒体。 它打开一个媒体容器,从中读取数据包,解码数据,然后将有关数据的信息分发到任何已注册的IMediaListener对象。 这是IMediaWriter类起作用的地方。 它对媒体进行编码和解码,同时处理音频和视频流。 为了使事情变得更有趣,我们还将IMediaViewer附加到我们的阅读器上。 它用作调试工具,使我们可以在解码视频时观看视频。 此外,在此过程中,我们还会看到各种统计信息。 请注意,该类处于实验模式,这意味着它有一些可能会挂起的错误,因此请谨慎处理。

本质上,使用上面的代码,我们将两个侦听器IMediaWriter和IMediaViewer附加到IMediaReader对象,并在读取器读取和解码源文件中的数据包的同时处理回调。 这是在“ while”循环中执行的。 如果我们使用示例输入文件运行该应用程序,将显示类似于以下的屏幕:

该过程完成后(由于我们正在同时实时查看,因此该过程将与源视频文件一样长),将以FLV格式创建一个新的输出文件。

让我们从命令行使用Ffmpeg来比较输入和输出文件:

ffmpeg.exe -ic:/myvideo.mp4
似乎流1编解码器的帧速率与容器的帧速率不同:59.92(14981/250)-> 29.96(14981/500)
从'c:/myvideo.mp4'输入#0,mov,mp4,m4a,3gp,3g2,mj2:
元数据:
major_brand:mp42
minor_version:0
兼容品牌:isomavc1mp42
持续时间:00:04:20.96,开始:0.000000,比特率:582 kb / s
流#0.0(und):音频:aac,44100 Hz,立体声,s16、115 kb / s
流#0.1(und):视频:h264,yuv420p,480×270 [PAR 1:1 DAR 16:9],464 kb / s,29.96 fps,29.96 tbr,29962 tbn,59.92 tbc

在原始视频文件中,容器为MPEG-4 ,有两个流:使用44100Hz的AAC的音频流和使用H.264的视频流。

ffmpeg.exe -ic:/myvideo.flv
似乎流0编解码器的帧速率与容器的帧速率不同:1000.00(1000/1)-> 29.97(30000/1001)
从'c:/myvideo.flv'输入#0 flv:
元数据:
持续时间:261
宽度:480
高度:270
videodatarate:62
帧率:30
videocodecid:2
音频数据率:62
音频采样率:44100
音频样本大小:16
立体声:真
音频编解码器:2
文件大小:43117478
持续时间:00:04:20.98,开始:0.000000,比特率:128 kb / s
流#0.0:视频:flv,yuv420p,480×270、64 kb / s,29.97 tbr,1k tbn,1k tbc
流#0.1:音频:mp3,44100 Hz,2声道,s16,64 kb / s

转码后,生成的Flash视频文件使用FLV视频流和MP3音频流。

现在,我们准备使用Xuggler修改媒体文件。 但是在编写代码之前,我们需要了解MediaTool的工作原理 :

MediaTool使用事件侦听器范例。 写入器会自动作为“侦听器”添加到读取器,并接收所有解码的媒体。 IMediaViewer和IMediaWriter接口(查看器和编写器实际上是什么)实现IMediaListener接口,并且可以作为侦听器添加到IMediaReader。

我们通过前面的示例确认了这一点。 问题是,为了对输入文件进行各种修改 ,我们必须建立一个“媒体管道”。 我们创建IMediaTool的自定义实现,然后以链式方式在每个工具上设置侦听器,以便它们将数据从一个传递到另一个。

假设我们希望在视频中添加静态图像,同时希望减少音频量。 在这种情况下,我们将创建两个自定义IMediaTool对象:

  • StaticImageMediaTool:拍摄视频图片并在屏幕上的特定位置标记静态图像文件。
  • VolumeAdjustMediaTool:按恒定因子调整音量。

另外,我们创建一个IMediaWriter对象,该对象将用于创建输出文件。 通过所有这些,我们创建了一条链,如下所示:

读取器-> addStaticImage-> reduceVolume->写入器

让我们看看实现以上所有功能的代码:

package com.javacodegeeks.xuggler;import java.awt.Color;
import java.awt.Graphics2D;
import java.awt.geom.Rectangle2D;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import java.nio.ShortBuffer;import javax.imageio.ImageIO;import com.xuggle.mediatool.IMediaReader;
import com.xuggle.mediatool.IMediaTool;
import com.xuggle.mediatool.IMediaWriter;
import com.xuggle.mediatool.MediaToolAdapter;
import com.xuggle.mediatool.ToolFactory;
import com.xuggle.mediatool.event.IAudioSamplesEvent;
import com.xuggle.mediatool.event.IVideoPictureEvent;public class ModifyMediaExample {private static final String inputFilename = "c:/myvideo.mp4";private static final String outputFilename = "c:/myvideo.flv";private static final String imageFilename = "c:/jcg_logo_small.png";public static void main(String[] args) {// create a media readerIMediaReader mediaReader = ToolFactory.makeReader(inputFilename);// configure it to generate BufferImagesmediaReader.setBufferedImageTypeToGenerate(BufferedImage.TYPE_3BYTE_BGR);IMediaWriter mediaWriter = ToolFactory.makeWriter(outputFilename, mediaReader);IMediaTool imageMediaTool = new StaticImageMediaTool(imageFilename);IMediaTool audioVolumeMediaTool = new VolumeAdjustMediaTool(0.1);// create a tool chain:// reader -> addStaticImage -> reduceVolume -> writermediaReader.addListener(imageMediaTool);imageMediaTool.addListener(audioVolumeMediaTool);audioVolumeMediaTool.addListener(mediaWriter);while (mediaReader.readPacket() == null) ;}private static class StaticImageMediaTool extends MediaToolAdapter {private BufferedImage logoImage;public StaticImageMediaTool(String imageFile) {try {logoImage = ImageIO.read(new File(imageFile));} catch (IOException e) {e.printStackTrace();throw new RuntimeException("Could not open file");}}@Overridepublic void onVideoPicture(IVideoPictureEvent event) {BufferedImage image = event.getImage();// get the graphics for the imageGraphics2D g = image.createGraphics();Rectangle2D bounds = new Rectangle2D.Float(0, 0, logoImage.getWidth(), logoImage.getHeight());// compute the amount to inset the time stamp and // translate the image to that positiondouble inset = bounds.getHeight();g.translate(inset, event.getImage().getHeight() - inset);g.setColor(Color.WHITE);g.fill(bounds);g.setColor(Color.BLACK);g.drawImage(logoImage, 0, 0, null);// call parent which will pass the video onto next tool in chainsuper.onVideoPicture(event);}}private static class VolumeAdjustMediaTool extends MediaToolAdapter {// the amount to adjust the volume byprivate double mVolume;public VolumeAdjustMediaTool(double volume) {mVolume = volume;}@Overridepublic void onAudioSamples(IAudioSamplesEvent event) {// get the raw audio bytes and adjust it's valueShortBuffer buffer = event.getAudioSamples().getByteBuffer().asShortBuffer();for (int i = 0; i < buffer.limit(); ++i) {buffer.put(i, (short) (buffer.get(i) * mVolume));}// call parent which will pass the audio onto next tool in chainsuper.onAudioSamples(event);}}}

与往常一样,我们首先创建一个IMediaReader并在调用IMediaListener.onVideoPicture时使用setBufferedImageTypeToGenerate方法生成BufferedImage图像。 为了将我们的自定义图像覆盖在实际的视频图片上,这是必要的。 然后,我们创建IMediaWriter和媒体工具对象,并如上所述配置工具链。 让我们仔细看看自定义媒体工具。

首先,我们有StaticImageMediaTool类。 它扩展了MediaToolAdapter并覆盖了onVideoPicture方法,因为我们希望使用这一方法来处理视频流。 在构造函数中,我们已经使用ImageIO.read方法加载了一个图像文件。 JavaCodeGeeks徽标用于此目的(实际上是其缩小版本)。 然后,在已实现的onVideoPicture方法中,我们通过调用IVideoPictureEvent.getImage获得基础的BufferedImage并创建一个Graphics2D对象。 然后,我们使用Graphics.drawImage方法覆盖静态图像。 最后,我们调用父级onVideoPicture方法,该方法会将视频传递到链中的下一个工具。

然后,我们有了VolumeAdjustMediaTool。 它还扩展了MediaToolAdapter ,但覆盖了onAudioSamples方法,该方法在对音频样本进行解码或编码后调用。 在那里,我们通过调用IAudioSamplesEvent.getAudioSamples来获取原始音频字节,并使用相应的ShortBuffer类调整其值。 再次,在自定义处理之后,我们调用父onAudioSamples方法,该方法会将音频传递到链中的下一个工具。 如果现在运行此应用程序,我们将在原始视频的顶部看到添加的图像,并且音频音量将大大降低。
而已。 Xuggler支持的转码和媒体处理。 与往常一样,您可以下载为本教程创建的Eclipse项目 。 请继续关注JavaCodeGeeks上的更多Xuggler教程! 别忘了分享!

相关文章:

  • Xuggler视频处理简介
  • Xuggler教程:帧捕获和视频创建
  • 使用wowza和xuggler将RTMP转换为RTSP

翻译自: https://www.javacodegeeks.com/2011/02/xuggler-tutorial-transcoding-media.html

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

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

相关文章

Web工程师必备的43款可视化工具

国外站点DATAVISUALIZATION.CH为大家总结出了当前热用的43款可视化工具&#xff0c;包括Arbor、Chroma.js、D3.js、Google Chart Tools等&#xff0c;绝对让你一饱眼福。 1.Arbor.js Arbor是一个免费的、可视化的图形库&#xff0c;基于矢量创建动态的连接图。它为图形组织和屏…

力扣7. 整数反转

方法一&#xff1a;官方给的&#xff0c;自己懂了后照着敲了一遍 class Solution {public int reverse(int x) {int rev 0;while (x ! 0) {//if判断条件过于复杂&#xff0c;我好不容易看懂了if (rev < Integer.MIN_VALUE / 10 || rev > Integer.MAX_VALUE / 10) {retu…

AndroidManifest.xml中的application中的name属性

被这个不起眼的属性折磨了一天&#xff0c;终于解决了。 由于项目需要&#xff0c;要合并两个android应用&#xff0c;于是拷代码&#xff0c;拷布局文件&#xff0c;拷values&#xff0c;所有的都搞定之后程序还是频频崩溃&#xff0c;一直没有找到原因&#xff0c;学android…

完美单身

我不时遇到一些Java程序员&#xff0c;他们不确定他们应该如何正确实现Singleton模式。 &#xff08;如果您不知道什么是Singleton&#xff0c;请尝试Wikipedia&#xff1a; Singleton模式 &#xff09;。 我并不是在谈论在线程环境中的正确实现。 但是&#xff0c;使用最常见的…

力扣移动零

给定一个数组 nums&#xff0c;编写一个函数将所有 0 移动到数组的末尾&#xff0c;同时保持非零元素的相对顺序。 我写的代码思路是从开始元素找0&#xff0c;找到了就将这个0后面的元素向前移一位 class Solution {public void moveZeroes(int[] nums) {int n nums.length;…

Doclava:来自Google的自定义Javadoc Doclet

Doclava是Google的自定义Javadoc Doclet&#xff0c;由Google Guice使用&#xff08;请参阅稍冷的 Javadocs &#xff09;。 Doclava使用JSilver作为其模板引擎&#xff0c;这是Clearsilver的纯Java实现。 我个人喜欢联合文档的想法&#xff0c;它允许文档在打开的项目上相互链…

linux cat显示若干行

【一】从第3000行开始&#xff0c;显示1000行。即显示3000~3999行 cat filename | tail -n 3000 | head -n 1000 【二】显示1000行到3000行 cat filename| head -n 3000 | tail -n 1000 *注意两种方法的顺序 分解&#xff1a; tail -n 1000&#xff1a;显示最后1000行 tail -n …

【Stackoverflow好问题】java在,如何推断阵列Array是否包括指定的值

问题java中&#xff0c;怎样推断数组Array是否包括指定的值精华回答1.Arrays.asList(...).contains(...) 2.使用 Apache Commons Lang包中的ArrayUtils.containsString[] fieldsToInclude { "id", "name", "location" };if ( ArrayUtils.contai…

力扣移除元素

给你一个数组 nums 和一个值 val&#xff0c;你需要 原地 移除所有数值等于 val 的元素&#xff0c;并返回移除后数组的新长度。 不要使用额外的数组空间&#xff0c;你必须仅使用 O(1) 额外空间并 原地 修改输入数组。 元素的顺序可以改变。你不需要考虑数组中超出新长度后面…

在域驱动设计中使用状态模式

域驱动设计&#xff08;DDD&#xff09;是一种软件开发方法&#xff0c;其中&#xff0c;通过将实现与核心业务概念的不断发展的模型相连接&#xff0c;可以解决问题的复杂性。 该术语是由Eric Evans创造的&#xff0c;并且有一个DDD专用站点可以促进其使用。 根据其定义&#…

使用selenium进行密码破解(绕过账号密码JS加密)

经常碰到网站&#xff0c;账号密码通过js加密后进行提交。通过burp拦截抓到的账号密码是加密后的&#xff0c;所以无法通过burp instruder进行破解。只能模拟浏览器填写表单并点击登录按钮进行破解。于是想到了自动化web测试工具selenium&#xff0c;代码如下&#xff0c;测试效…

力扣删除排序数组中的重复项

给你一个有序数组 nums &#xff0c;请你 原地 删除重复出现的元素&#xff0c;使每个元素 只出现一次 &#xff0c;返回删除后数组的新长度。 不要使用额外的数组空间&#xff0c;你必须在 原地 修改输入数组 并在使用 O(1) 额外空间的条件下完成。 我没注意到“有序”这一条…

POJ1789-Truck History .

题目链接&#xff1a;http://poj.org/problem?id1789 题目的大概意思就是给你n个字符串。每个字符串只有7的长度。然后分别给这些字符串编号。不同编号之间的距离就是他们有多少个不同的字母。&#xff08;同一个位置字母不相同也算&#xff09;然后一个编号只能由另一个派生…

Java Fork / Join进行并行编程

最近几年&#xff0c;计算机处理器领域发生了范式转变。 多年来&#xff0c;处理器制造商一直在提高时钟频率&#xff0c;因此开发人员享受到这样的事实&#xff0c;即他们的单线程软件执行得更快&#xff0c;而无需他们付出任何努力。 现在&#xff0c;处理器制造商青睐多核芯…

arm-elf-gcc交叉编译器的使用教程

arm-elf-gcc交叉编译器的使用教程 一开始需要安装arm-elf-gcc&#xff0c;但是这是一个32位的程序&#xff0c;我是安装了64位的系统&#xff0c;据说安装ia32.libs依赖库能运行这个&#xff0c;但是看到博客上面前人安装完了系统图标少了一半&#xff0c;然后就怕了。经过了翻…

力扣删除排序数组中的重复项 II

给你一个有序数组 nums &#xff0c;请你 原地 删除重复出现的元素&#xff0c;使每个元素 最多出现两次 &#xff0c;返回删除后数组的新长度。 不要使用额外的数组空间&#xff0c;你必须在 原地 修改输入数组 并在使用 O(1) 额外空间的条件下完成。 思路&#xff1a; 双指针…

2 android学习资料

http://blog.csdn.net/lmj623565791 http://blog.csdn.net/harvic880925/article/details/50995268转载于:https://www.cnblogs.com/YyuTtian/p/5440930.html

建立自己的GWT Spring Maven原型

大家好&#xff0c; 在观看Justin撰写的有关Spring和GWT的非常有趣的文章时&#xff0c;我认为展示如何构建自己的自定义Maven原型非常有用。我们将展示的原型基于Justin的上一个项目&#xff0c;并包括各种技术&#xff0c;例如Spring &#xff0c; GWT &#xff0c; AspectJ…

C# 连接Oracle数据库异常总结

这2天因为工作需要连接Oracle数据库&#xff0c;中间发生了很多问题 一、使用OleDbConnection连接数据库 ------------------ ProviderOraOLEDB.Oracle.1;User IDsajet;Passwordtech;Data Source(DESCRIPTION (ADDRESS_LIST (ADDRESS (PROTOCOL TCP)(HOST 192.168.66.225)(…

力扣颜色分类

给定一个包含红色、白色和蓝色&#xff0c;一共 n 个元素的数组&#xff0c;原地对它们进行排序&#xff0c;使得相同颜色的元素相邻&#xff0c;并按照红色、白色、蓝色顺序排列。 此题中&#xff0c;我们使用整数 0、 1 和 2 分别表示红色、白色和蓝色。 思路:将红色和蓝色…