JavaCV:将任意带旋转角度信息视频转码为h264编码的mp4格式视频

前言

最近发现之前写的博客JavaCV:将任意视频转码为h264编码的mp4格式视频不能正确处理微信上传的视频。主要表现为明明是手机竖屏拍摄的视频,转码后的却是横屏。通过查看程序输出提示:

Info:     Side data:

Info:       
Info: displaymatrix: rotation of -180.00 degrees
Info: 

可看到这部分视频带有额外的显示矩阵数据(即旋转角度),为此在原博客代码上增加视频旋转画面处理。

视频工具类

import static org.bytedeco.opencv.global.opencv_core.CV_8UC3;
import static org.bytedeco.opencv.global.opencv_core.ROTATE_180;
import static org.bytedeco.opencv.global.opencv_core.ROTATE_90_CLOCKWISE;
import static org.bytedeco.opencv.global.opencv_core.ROTATE_90_COUNTERCLOCKWISE;
import static org.bytedeco.opencv.global.opencv_core.rotate;

import org.bytedeco.ffmpeg.avcodec.AVCodecParameters;
import org.bytedeco.ffmpeg.avformat.AVFormatContext;
import org.bytedeco.ffmpeg.avformat.AVStream;
import org.bytedeco.ffmpeg.global.avcodec;
import org.bytedeco.javacv.FFmpegFrameGrabber;
import org.bytedeco.javacv.FFmpegFrameRecorder;
import org.bytedeco.javacv.FFmpegLogCallback;
import org.bytedeco.javacv.Frame;
import org.bytedeco.javacv.FrameGrabber;
import org.bytedeco.javacv.OpenCVFrameConverter;
import org.bytedeco.opencv.opencv_core.Mat;
import org.bytedeco.opencv.opencv_core.UMat;

/**
 * 视频工具类
 * 
 * @author alderaan
 * @version 创建时间:2023年6月20日 下午3:31:22
 *
 */
public class VideoUtil
{
	private static OpenCVFrameConverter.ToMat converter = new OpenCVFrameConverter.ToMat();	
	
	/**
	 * 视频转码函数(仅转码)
	 * 
	 * @param inputfile  原始视频文件完整路径
	 * @param outputfile 目标视频文件完整保存路径(必须完整文件名,即包含格式后缀,推荐格式后缀为.mp4)
	 * @throws Exception 异常
	 */
	public static void videoConvert(String inputfile, String outputfile) throws Exception
	{
		if (outputfile.lastIndexOf('.') < 0)
		{
			throw new Exception("Error! Output file format undetected!");
		}
		String format = outputfile.substring(outputfile.lastIndexOf('.'));

		FFmpegLogCallback.set();
		Frame frame;
		FFmpegFrameGrabber grabber = new FFmpegFrameGrabber(inputfile);
		FFmpegFrameRecorder recorder = null;

		try
		{
			System.out.println("开始初始化帧抓取器");

			// 初始化帧抓取器,例如数据结构(时间戳、编码器上下文、帧对象等),
			// 如果入参等于true,还会调用avformat_find_stream_info方法获取流的信息,放入AVFormatContext类型的成员变量oc中
			grabber.start(true);

			System.out.println("帧抓取器初始化完成");

			// grabber.start方法中,初始化的解码器信息存在放在grabber的成员变量oc中
			AVFormatContext avformatcontext = grabber.getFormatContext();

			// 文件内有几个媒体流(一般是视频流+音频流)
			int streamNum = avformatcontext.nb_streams();

			// 没有媒体流就不用继续了
			if (streamNum < 1)
			{
				System.out.println("文件内不存在媒体流");
				throw new Exception("Error! There is no media stream in the file!");
			}

			// 取得视频的帧率
			double framerate = grabber.getVideoFrameRate();

			System.out.printf("视频帧率[%f],视频时长[%d]秒,媒体流数量[%d]\r\n", framerate, avformatcontext.duration() / 1000000,
					avformatcontext.nb_streams());

			// 遍历每一个流,检查其类型
			for (int i = 0; i < streamNum; i++)
			{
				AVStream avstream = avformatcontext.streams(i);
				AVCodecParameters avcodecparameters = avstream.codecpar();
				System.out.printf("流的索引[%d],编码器类型[%d],编码器ID[%d]\r\n", i, avcodecparameters.codec_type(),
						avcodecparameters.codec_id());
			}

			// 视频宽度
			int frameWidth = grabber.getImageWidth();
			// 视频高度
			int frameHeight = grabber.getImageHeight();
			// 音频通道数量
			int audioChannels = grabber.getAudioChannels();

			double rotate = grabber.getDisplayRotation();

			System.out.printf("旋转角度[%s],视频宽度[%d],视频高度[%d],音频通道数[%d]\r\n", rotate, frameWidth, frameHeight,
					audioChannels);

			// 目标视频宽度
			int dstFrameWidth = 0;
			// 目标视频高度
			int dstFrameHeight = 0;

			int rotate_int = (int) Math.abs(rotate);
			if (rotate_int == 90 || rotate_int == 270)
			{
				dstFrameWidth = frameHeight;
				dstFrameHeight = frameWidth;
			} else
			{
				dstFrameWidth = frameWidth;
				dstFrameHeight = frameHeight;
			}
			
			recorder = new FFmpegFrameRecorder(outputfile, dstFrameWidth, dstFrameHeight, audioChannels);
			recorder.setVideoCodec(avcodec.AV_CODEC_ID_H264);

			recorder.setFormat(format);
			// 使用原始视频的码率,若需要则自行修改码率
			recorder.setVideoBitrate(grabber.getVideoBitrate());
			
			// 一秒内的帧数,帧率
			recorder.setFrameRate(framerate);

			// 两个关键帧之间的帧数
			recorder.setGopSize((int)framerate);

			// 设置音频通道数,与视频源的通道数相等
			recorder.setAudioChannels(grabber.getAudioChannels());

			recorder.start();

			int videoframenum = 0;
			int audioframenum = 0;
			int dataframenum = 0;

			UMat umat = new UMat(frameHeight, frameWidth, CV_8UC3);
			UMat umatDst = new UMat(dstFrameHeight, dstFrameWidth, CV_8UC3);
			Mat matDst = new Mat(dstFrameHeight, dstFrameWidth, CV_8UC3);
			
			// 持续从视频源取帧
			while (null != (frame = grabber.grab()))
			{
				// 有图像,就把视频帧加一
				if (null != frame.image)
				{
					videoframenum++;
					
					if (rotate_int != 0)
					{
						Mat mat = converter.convertToMat(frame);
						mat.copyTo(umat);
						if (rotate == -90 || rotate == 270)
						{
							rotate(umat, umatDst, ROTATE_90_CLOCKWISE);
						} else if (rotate == -180 || rotate == 180)
						{
							rotate(umat, umatDst, ROTATE_180);
						} else if (rotate == 90 || rotate == -270)
						{
							rotate(umat, umatDst, ROTATE_90_COUNTERCLOCKWISE);
						}
						umatDst.copyTo(matDst);
						Frame frame_temp = converter.convert(matDst);
						recorder.record(frame_temp);
					} else
					{
						// 取出的每一帧,都保存到视频
						recorder.record(frame);
					}
				}

				// 有声音,就把音频帧加一
				if (null != frame.samples)
				{
					audioframenum++;
					// 取出的每一帧,都保存到视频
					recorder.record(frame);
				}

				// 有数据,就把数据帧加一
				if (null != frame.data)
				{
					dataframenum++;
				}
			}

			System.out.printf("转码完成,视频帧[%d],音频帧[%d],数据帧[%d]\r\n", videoframenum, audioframenum, dataframenum);

		} catch (Exception e)
		{
			// e.printStackTrace();
			throw e;
		} finally
		{
			if (recorder != null)
			{
				try
				{
					recorder.close();
				} catch (Exception e)
				{
					// System.out.println("recorder.close异常" + e);
					throw e;
				}
			}

			try
			{
				grabber.close();
			} catch (FrameGrabber.Exception e)
			{
				// System.out.println("frameGrabber.close异常" + e);
				throw e;
			}
		}
	}
}


测试类

/**
 * 测试类
 * 
 * @author alderaan
 * @version 创建时间:2023年6月20日 下午3:32:09
 *
 */
public class App
{
	public static void main(String[] args)
	{
		try
		{
			// videoConvert函数,根据outputfile的格式后缀设置转码后的视频格式,推荐使用mp4格式后缀
			VideoUtil.videoConvert("/home/alderaan/t1.mp4", "/home/alderaan/result.mp4");
		} catch (java.lang.Exception e)
		{
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		System.out.println("执行完毕!");
	}
}

POM

需要升级javacv版本到1.5.8

		<dependency>
			<groupId>org.bytedeco</groupId>
			<artifactId>javacv</artifactId>
			<version>1.5.8</version>
		</dependency>
		<!-- Linux x86_64 使用 -->
		<dependency>
			<groupId>org.bytedeco</groupId>
			<artifactId>javacpp</artifactId>
			<version>1.5.8</version>
		</dependency>
		<dependency>
			<groupId>org.bytedeco</groupId>
			<artifactId>opencv</artifactId>
			<version>4.6.0-1.5.8</version>
			<classifier>linux-x86_64-gpu</classifier>
		</dependency>
		<dependency>
			<groupId>org.bytedeco</groupId>
			<artifactId>openblas</artifactId>
			<version>0.3.21-1.5.8</version>
			<classifier>linux-x86_64</classifier>
		</dependency>
		<dependency>
			<groupId>org.bytedeco</groupId>
			<artifactId>ffmpeg</artifactId>
			<version>5.1.2-1.5.8</version>
			<classifier>linux-x86_64-gpl</classifier>
		</dependency>
		<!-- Windows x86_64 使用 -->
		<dependency>
			<groupId>org.bytedeco</groupId>
			<artifactId>opencv</artifactId>
			<version>4.6.0-1.5.8</version>
			<classifier>windows-x86_64</classifier>
		</dependency>
		<dependency>
			<groupId>org.bytedeco</groupId>
			<artifactId>openblas</artifactId>
			<version>0.3.21-1.5.8</version>
			<classifier>windows-x86_64</classifier>
		</dependency>
		<dependency>
			<groupId>org.bytedeco</groupId>
			<artifactId>ffmpeg</artifactId>
			<version>5.1.2-1.5.8</version>
			<classifier>windows-x86_64</classifier>
		</dependency>
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值