视频操作类

接 视频分割项目预研
http://zhuyufufu.iteye.com/blog/2078404

   对于没法用ffmpeg处理的rm等格式,使用了 mplayermencoder来处理。

   我的项目重点在视频分割。在此只写rmvb切分的代码。对于别的 ffmpeg无法处理的格式, mencoder应该都能处理。

   下面贴出暂时使用的视频处理工具类,有获 取视频信息、截取视频、获取截图、加水印、视频转换等功能。

   其原理都很简单: 使用java调用外部组件对视频进行处理

   后面要添加的功能: 解析组件输出流,生成视频处理进度及结果

下面贴上代码,方便以后查阅

properties配置文件
#视频切割配置参数
ffmpeg_home=D:/ffmpeg/ffmpeg-20140611-git-b2fb65c-win64-static/
ffmpeg=D:/ffmpeg/ffmpeg-20140611-git-b2fb65c-win64-static/bin/ffmpeg.exe
mplayer=D:/ffmpeg/MPlayer-generic-r37220+gd4be3a8/mplayer.exe
mencoder=D:/ffmpeg/MPlayer-generic-r37220+gd4be3a8/mencoder.exe


package com.zas.util;

import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;

import org.apache.log4j.Logger;

/**
 * 视频操作类
 * @author zas
 * 
 */
public class VideoUtil {
	static Logger logger = Logger.getLogger(VideoUtil.class);

	final static String FFMPEG = PropertyToolkits.getProperty("ffmpeg");
	final static String MENCODER = PropertyToolkits.getProperty("mencoder");
	final static String MPLAYER = PropertyToolkits.getProperty("mplayer");
	
	/**
	 * 获取一个指定视频的基本信息
	 * @param inputVideoFile
	 * @return 视频信息
	 */
	public static String getVideoInfo(String inputVideoFile){
		//检查是否能够处理
		if(!checkVideoFile(inputVideoFile)){
			throw new RuntimeException("文件有误");
		}
		List<String> commandList = new ArrayList<String>();
		commandList.add(FFMPEG);
		commandList.add("-i");
		commandList.add(inputVideoFile);
		
//		mplayer -identify $inputFileName -nosound -vc dummy -vo null
//		List<String> commandList = new ArrayList<String>();
//		commandList.add(MPLAYER);
//		commandList.add("-identify");
//		commandList.add(inputVideoFile);
//		commandList.add("-nosound");
//		commandList.add("-vc");
//		commandList.add("dummy");
//		commandList.add("-vo");
//		commandList.add("null");
		
		//获取视频信息
		String videoInfo = process(commandList);
				
		return videoInfo;
	}
	
	
	/**
	 * 获取视频截图
	 * @param inputVideoFile 要处理的视频
	 * @param imgPath 要截取的截图
	 * @param parameterMap 参数
	 * @return
	 */
	public static String getVideoSnapshots(String inputVideoFile, String imgPath, Map<String, String> parameterMap){
		//ffmpeg -i 1111.wma -y -ss 00:00:09 -t 00:00:10 -s 320*240 -f mjpeg -vframes 10 1111_1.jpg
		//获取图片的第一帧 ffmpeg commandLine: ffmpeg -y -i 1111.wma -vframes  1 -r 1 -ac 1 -ab 2 -s 320x240 -f image2 1111_1.jpg
		//把视频的前30帧转换成一个Animated Gif :  
		//ffmpeg -i 1111.wma -vframes 30 -y -f gif 1111.gif
		
		//图片时间截取也很重要,很有可能是无效图片或者是黑屏
		//建议 增加关键帧,通常第一帧为关键帧,可以使用:vframes:帧参数,舍弃微秒参数,只保留时间参数

		//检查是否能够处理
		if(!checkVideoFile(inputVideoFile)){
			throw new RuntimeException("视频文件有误");
		}
		
		/*
		 	-i filename 输入文件
			-y 覆盖输出文件
			-t duration 设置纪录时间 hh:mm:ss[.xxx]格式的记录时间也支持
			-ss position 搜索到指定的时间 [-]hh:mm:ss[.xxx]的格式也支持
		 */
		List<String> commandList = new ArrayList<String>();
		commandList.add(FFMPEG);
		commandList.add("-i");
		commandList.add(inputVideoFile);
		commandList.add("-y");
		//位置参数太靠后,会影响抓图效率
		commandList.add("-ss");
		commandList.add("00:00:09");
//		commandList.add("-t");
//		commandList.add("00:00:01");
		commandList.add("-vframes");
		commandList.add("1");
//		commandList.add("-r");
//		commandList.add("1");
//		commandList.add("-ac");
//		commandList.add("1");
//		commandList.add("-ab");
//		commandList.add("2");
//		commandList.add("-s");
//		commandList.add("320*240");
		commandList.add("-f");
		commandList.add("mjpeg");
		commandList.add(imgPath);
		
		String processInfo = process(commandList);
		return processInfo;
	}
	
	/**
	 * 使用ffmpeg 从指定时间开始截取特定时长视频
	 * @param fromTime 开始时间 	00:5:28
	 * @param inputVideoFile 要截取的视频
	 * @param duration 要截取的视频时长	00:03:25
	 * @param outputFile 截取的视频的输出位置
	 * @param parameterMap 参数
	 * @return 处理信息
	 */
	public static String cutting(String fromTime, String inputVideoFile, String duration, String outputFile, Map<String, String> parameterMap){
		//检查是否能够处理
		if(!checkVideoFile(inputVideoFile)){
			throw new RuntimeException("视频文件有误");
		}
		File file = new File(outputFile);
		if(file.exists()){
			System.out.println("outputFile exists !");
			return "outputFile exists!";
		}
		String filetype = inputVideoFile.substring(inputVideoFile.lastIndexOf(".") + 1);
		if("rm".equalsIgnoreCase(filetype) || "rmvb".equalsIgnoreCase(filetype)){
			return cuttingRmvb(fromTime, inputVideoFile, duration, outputFile, parameterMap);
		}
		//ffmpeg -ss 00:5:28 -i "1111.wmv" -acodec copy -vcodec copy -t 00:03:25 output.wmv 
		//这行命令解释为:从文件 1111.wmv 第 5:28 分秒开始,截取 03: 25 的时间,其中视频和音频解码不变,输出文件名为 output.wmv 。 
		
		List<String> commandList = new ArrayList<String>();
		commandList.add(FFMPEG);
		commandList.add("-ss");
		commandList.add(fromTime);
		commandList.add("-i");
		commandList.add(inputVideoFile);
		commandList.add("-acodec");
		commandList.add("copy");
		commandList.add("-vcodec");
		commandList.add("copy");
		commandList.add("-t");
		commandList.add(duration);
		commandList.add(outputFile);
		
		String processInfo = process(commandList);
		return processInfo;
	}
	
	/**
	 * 使用ffmpeg 从指定时间开始截取特定时长视频
	 * @param fromTime 开始时间 	00:5:28
	 * @param inputVideoFile 要截取的视频
	 * @param duration 要截取的视频时长	00:03:25
	 * @param outputFile 截取的视频的输出位置
	 * @param parameterMap 参数
	 * @return 处理信息
	 */
	public static String cuttingRmvb(String fromTime, String inputVideoFile, String duration, String outputFile, Map<String, String> parameterMap){
		//检查是否能够处理
		if(!checkVideoFile(inputVideoFile)){
			throw new RuntimeException("视频文件有误");
		}
//		mencoder basket.rm -ovc lavc -oac mp3lame -o basket.avi -ss 5:00 -endpos 8:00
		List<String> commandList = new ArrayList<String>();
		commandList.add(MENCODER);
		commandList.add(inputVideoFile);
		commandList.add("-ovc");
		commandList.add("lavc");
		commandList.add("-oac");
		commandList.add("mp3lame");
		commandList.add("-ss");
		commandList.add(fromTime);
		commandList.add("-endpos");
		commandList.add(duration);
		commandList.add("-o");
		commandList.add(outputFile);
		
		String processInfo = process(commandList);
		return processInfo;
	}

	/**
	 * 视频转换
	 * @param inputVideoFile 视频文件 
	 * @param outputFile 转换后的视频文件
	 * @param parameterMap 其余参数
	 * @return
	 */
	public static String videoConverter(String inputVideoFile, String outputFile, Map<String, String> parameterMap){
		//检查是否能够处理
		if(!checkVideoFile(inputVideoFile)){
			throw new RuntimeException("视频文件有误");
		}
		//视频的格式有很多,以mp4和flv为例子
		// ffmpeg -i test.mp4 -ab 56 -ar 22050 -qmin 2 -qmax 16 -b 320k -r 15 -s 320x240 outputfile.flv   //mp4 转 flv
		// ffmpeg -i outputfile.flv -copyts -strict -2 test.mp4  //flv 转 mp4
		
		List<String> commandList = new ArrayList<String>();
        commandList.add(FFMPEG);
        commandList.add("-y");
        commandList.add("-i");
        commandList.add(inputVideoFile);
        commandList.add("-ab");
        commandList.add("5600000");
        commandList.add("-ar");
        commandList.add("22050");
        commandList.add("-b");
        commandList.add("500");
        commandList.add("-s");
        commandList.add("320*240");
        commandList.add(outputFile);
		String processInfo = process(commandList);
		return processInfo;
	}
	
	/**
	 * 视频加水印
	 * @param inputVideoFile 要处理的视频文件
	 * @param outputFile	输出的视频文件
	 * @param parameterMap 参数
	 * @return
	 */
	public static String addWatermark(String inputVideoFile, String outputFile, Map<String, String> parameterMap){
		//检查是否能够处理
		if(!checkVideoFile(inputVideoFile)){
			throw new RuntimeException("视频文件有误");
		}
		//http://blog.51yip.com/linux/1584.html
		//#ffmpeg -y -i test.mp4 -acodec copy -vf "movie=logo.jpg [logo]; [in][logo] overlay=10:10:1 [out]" test2.mp4
		//overlay=10:10:1,后三个数据表示是距离左边的距离,距离上边的距离,是否透明,1表示透明。上例我用的是jpg,当然不可能透明。
		//# ffmpeg -y -i test.mp4 -acodec copy -vf "movie=uwsgi.jpg [logo]; [in][logo] overlay=enable='lte(t,1)' [out]" test2.mp4
		//overlay=enable='lte(t,1)' ,这个参数表示,水印在前一秒显示。
		
		//ffmpeg -y -i D:/ffmpeg/video/w.mkv -acodec copy -t 00:01:10 -vf "movie=logo.png [logo]; [in][logo] overlay=10:10:1 [out]" D:/ffmpeg/video/w_logo.mkv
		List<String> commandList = new ArrayList<String>();
		commandList.add(FFMPEG);
		commandList.add("-i");
		commandList.add(inputVideoFile);
		commandList.add("-y");
		commandList.add("-acodec");
		commandList.add("copy");
		commandList.add("-vf");
		commandList.add("\"movie=logo.png [logo]; [in][logo] overlay=10:10:1 [out]\"");
		commandList.add(outputFile);
//		
//		String command = FFMPEG + " -y -i \"D:/ffmpeg/video/a b/w.mkv\" -acodec copy -t 00:01:10 -vf \"movie=logo.png [logo]; [in][logo] overlay=10:10:1 [out]\" D:/ffmpeg/video/w_logo.mkv";
//		String processInfo = exec(command);
		String processInfo = process(commandList);
		return processInfo;
	}
	
	/**
	 * 命令执行
	 * @param command
	 * @return
	 */
	public static String exec(String command) {
		long beginTime = System.nanoTime();
		Runtime rt = Runtime.getRuntime();
		try {
			Process process = rt.exec(command);
			
		/*	StringTokenizer st = new StringTokenizer(command);
			String[] cmd = new String[st.countTokens()];
		 	for (int i = 0; st.hasMoreTokens(); i++){
		 		cmd[i] = st.nextToken();
		 		System.out.println(cmd[i]);
		 	}
		 	cmd[0] = new File(cmd[0]).getPath();

			StringBuilder cmdbuf = new StringBuilder(80);
			for (int i = 0; i < cmd.length; i++) {
				if (i > 0) {
					cmdbuf.append(' ');
				}
				String s = cmd[i];
				if (s.indexOf(' ') >= 0 || s.indexOf('\t') >= 0) {
					if (s.charAt(0) != '"') {
						cmdbuf.append('"');
						cmdbuf.append(s);
						if (s.endsWith("\\")) {
							cmdbuf.append("\\");
						}
						cmdbuf.append('"');
					} else if (s.endsWith("\"")) {
						 The argument has already been quoted. 
						cmdbuf.append(s);
					} else {
						 Unmatched quote for the argument. 
						throw new IllegalArgumentException();
					}
				} else {
					cmdbuf.append(s);
				}
			}
			String cmdstr = cmdbuf.toString();
			System.out.println("cmdstr : " + cmdstr);*/
			final InputStream isNormal = process.getInputStream();
			new Thread(new Runnable() {
			    public void run() {
			        BufferedReader br = new BufferedReader(new InputStreamReader(isNormal)); 
			        StringBuilder buf = new StringBuilder();
					String line = null;
					try {
						while((line = br.readLine()) != null){
							buf.append(line + "\n");
						}
					} catch (IOException e) {
						e.printStackTrace();
					}
					System.out.println("输出结果为:" + buf);
			    }
			}).start(); // 启动单独的线程来清空process.getInputStream()的缓冲区
			
			InputStream isError = process.getErrorStream();
			BufferedReader br2 = new BufferedReader(new InputStreamReader(isError)); 
			StringBuilder buf = new StringBuilder();
			String line = null;
			while((line = br2.readLine()) != null){
				buf.append(line + "\n");
			}
			System.out.println("错误输出结果为:" + buf);
			
			try {
				process.waitFor();
			} catch (InterruptedException e) {
				e.printStackTrace();
			}

		} catch (IOException e) {
			e.printStackTrace();
		}
		long endTime = System.nanoTime();
		System.out.println("视频处理耗时: " + (endTime - beginTime) / 1000000 + " 毫秒 ");
		return null;
	}


	/**
	 * 根据命令处理视频
	 * @param commandList
	 * @param processInfo
	 * @return 处理信息
	 */
	private static String process(List<String> commandList) {
		StringBuffer processInfo = new StringBuffer();
		ProcessBuilder builder = new ProcessBuilder();
		builder.command(commandList);
		builder.redirectErrorStream(true);
		long beginTime = System.nanoTime();
		try {
			Process p = builder.start();
			//保存ffmpeg的输出结果流
			BufferedReader buf = null; 
			buf = new BufferedReader(new InputStreamReader(p.getInputStream()));
			
			String line = null;
			while ((line = buf.readLine()) != null) {
				System.out.println(line);
				processInfo.append(line + "\n");
			}
			p.waitFor();// 这里线程阻塞,将等待外部转换进程运行成功运行结束后,才往下执行

		} catch (IOException e) {
			e.printStackTrace();
			logger.error(e);
			throw new RuntimeException("视频处理出错");
		} catch (InterruptedException e) {
			e.printStackTrace();
			logger.error(e);
			throw new RuntimeException("视频处理出错");
		}
		long endTime = System.nanoTime();
		System.out.println("处理耗时: " + (endTime - beginTime) / 1000000 + " 毫秒。 ");
		System.out.println("视频处理结果信息: \n" + processInfo);
		return processInfo.toString();
	}

	/**
	 * 检测视频是否能够被处理
	 * @param videoPath
	 * @return
	 */
	private static boolean checkVideoFile(String videoPath) {
		if(null == videoPath){
			return false;
		}
		//根据后缀做类型检测
		
		//检查是否文件以及文件是否存在
		File videoFile = new File(videoPath);
		if(!videoFile.isFile() || !videoFile.exists()){
			return false;
		}
		
		return true;
	}

	public static void main(String[] args) {
//		String inputVideoFile = "D:/ffmpeg/video/【天下足球网www.txzqw.com】下半场.rmvb";
		String outputFile = "D:/ffmpeg/video/【天下足球网www.txzqw.com】下半场_1.rmvb";
		String inputVideoFile = "D:/ffmpeg/video/a b/w.mkv";
//		String outputFile = "D:/ffmpeg/video/w_1.mkv";
//		String inputVideoFile = "D:/ffmpeg/video/Wildlife.wmv";
//		String outputFile = "D:/ffmpeg/video/Wildlife.wmv.flv";
		String fromTime = "00:00:00";
		String duration = "00:3:28";
		VideoUtil.getVideoInfo(inputVideoFile);
//		VideoUtil.cuttingRmvb(fromTime, inputVideoFile, duration, outputFile, null);
//		String imgPath = "D:/ffmpeg/video/1111.jpg";
//		VideoUtil.getVideoSnapshots(inputVideoFile, imgPath, null);
//		VideoUtil.addWatermark(inputVideoFile, outputFile, null);
//		VideoUtil.videoConverter(inputVideoFile, outputFile, null);
//		VideoUtil.getVideoInfo(inputVideoFile);
	}

}

  

你可能感兴趣的:(java,ffmpeg,mencoder,mplayer)