Java使用ffmpeg进行视频格式转换、音视频合并、播放、截图

上一篇进行了ffmpeg的下载及安装,在下面有安装步骤

本篇在上一篇的基础上进行代码封装,Java里使用ProcessBuilder模拟命令行调用ffmpeg进行视频格式转换、音视频合并、播放、截图。
需先安装ffmpeg

FfmpegUtils封装类:
ffplay、ffmpeg、ffprobe是安装的ffmpeg路径。

import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStreamReader;
import java.time.LocalDate;
import java.time.LocalTime;
import java.time.format.DateTimeFormatter;
import java.util.ArrayList;
import java.util.List;
import java.util.regex.Matcher;
import java.util.regex.Pattern;import static java.time.temporal.ChronoUnit.*;/*** 利用ffmpeg进行音频视频操作,需先下载安装ffmpeg*/
public class FfmpegUtils {/*** 下载的ffmpeg解压后的bin目录路径,可配置到环境变量通过配置文件读取*/private static String ffplay = "D:\\Program File\\ffmpeg-4.3.1-2021-01-01-essentials_build\\bin\\ffplay.exe";private static String ffmpeg = "D:\\Program File\\ffmpeg-4.3.1-2021-01-01-essentials_build\\bin\\ffmpeg.exe";private static String ffprobe = "D:\\Program File\\ffmpeg-4.3.1-2021-01-01-essentials_build\\bin\\ffprobe.exe";/*** 提取的音频、合成的视频存放路径,不存在会自动创建*/private static String saveMediaPath = "D:\\ffmpegMedia\\";/*** 保存音频、视频的临时文件夹,不存在会自动创建*/private static String tempMediaPath = "D:\\ffmpegMedia\\temp\\";/*** 保存图片截图的文件夹,不存在会自动创建*/private static String picturMediaPath = "D:\\ffmpegMedia\\pictur\\";static {//如果没有文件夹,则创建File saveMediaFile = new File(saveMediaPath);if (!saveMediaFile.exists() && !saveMediaFile.isDirectory()) {saveMediaFile.mkdirs();}File tempMediaFile = new File(tempMediaPath);if (!tempMediaFile.exists() && !tempMediaFile.isDirectory()) {tempMediaFile.mkdirs();}File picturMediaFile = new File(picturMediaPath);if (!picturMediaFile.exists() && !picturMediaFile.isDirectory()) {picturMediaFile.mkdirs();}}/*** 播放音频和视频** @param resourcesPath 文件的路径*/public static void playVideoAudio(String resourcesPath) {List<String> command = new ArrayList<>();command.add(ffplay);command.add("-window_title");String fileName = resourcesPath.substring(resourcesPath.lastIndexOf("\\") + 1);command.add(fileName);command.add(resourcesPath);//播放完后自动退出//command.add("-autoexit");commandStart(command);}/*** 播放音频和视频并指定循环次数** @param resourcesPath 文件的路径* @param loop          循环播放次数*/public static void playVideoAudio(String resourcesPath, int loop) {List<String> command = new ArrayList<>();command.add(ffplay);command.add("-window_title");String fileName = resourcesPath.substring(resourcesPath.lastIndexOf("\\") + 1);command.add(fileName);command.add(resourcesPath);command.add("-loop");command.add(String.valueOf(loop));//播放完后自动退出//command.add("-autoexit");commandStart(command);}/*** 播放音频和视频并指定宽、高、循环次数** @param resourcesPath 文件的路径* @param weight        宽度* @param height        高度* @param loop          循环播放次数*/public static void playVideoAudio(String resourcesPath, int weight, int height, int loop) {List<String> command = new ArrayList<>();command.add(ffplay);command.add("-window_title");String fileName = resourcesPath.substring(resourcesPath.lastIndexOf("\\") + 1);command.add(fileName);command.add(resourcesPath);command.add("-x");command.add(String.valueOf(weight));command.add("-y");command.add(String.valueOf(height));command.add("-loop");command.add(String.valueOf(loop));//播放完后自动退出//command.add("-autoexit");commandStart(command);}/*** 调用命令行执行** @param command 命令行参数*/public static void commandStart(List<String> command) {command.forEach(v -> System.out.print(v + " "));System.out.println();System.out.println();ProcessBuilder builder = new ProcessBuilder();//正常信息和错误信息合并输出builder.redirectErrorStream(true);builder.command(command);//开始执行命令Process process = null;try {process = builder.start();//如果你想获取到执行完后的信息,那么下面的代码也是需要的String line = "";BufferedReader br = new BufferedReader(new InputStreamReader(process.getInputStream()));while ((line = br.readLine()) != null) {System.out.println(line);}} catch (IOException e) {e.printStackTrace();}}/*** 调用命令行执行,并返回信息** @param command 命令行参数*/public static String getInfoStr(List<String> command) {command.forEach(v -> System.out.print(v + " "));System.out.println();System.out.println();ProcessBuilder builder = new ProcessBuilder();//正常信息和错误信息合并输出builder.redirectErrorStream(true);builder.command(command);//开始执行命令Process process = null;StringBuffer sb = new StringBuffer();try {process = builder.start();//如果你想获取到执行完后的信息,那么下面的代码也是需要的String line = "";BufferedReader br = new BufferedReader(new InputStreamReader(process.getInputStream()));while ((line = br.readLine()) != null) {System.out.println(line);sb.append(line);}} catch (IOException e) {e.printStackTrace();}return String.valueOf(sb);}/*** 从视频中提取音频为mp3** @param videoResourcesPath 视频文件的路径*/public static void getAudioFromVideo(String videoResourcesPath) {String fileName = videoResourcesPath.substring(videoResourcesPath.lastIndexOf("\\") + 1, videoResourcesPath.lastIndexOf("."));List<String> command = new ArrayList<>();command.add(ffmpeg);command.add("-i");command.add(videoResourcesPath);command.add(saveMediaPath + fileName + ".mp3");commandStart(command);}/*** 从视频中去除去音频并保存视频** @param videoResourcesPath 视频文件的路径*/public static void getVideoFromAudio(String videoResourcesPath) {List<String> command = new ArrayList<>();command.add(ffmpeg);command.add("-i");command.add(videoResourcesPath);command.add("-vcodec");command.add("copy");command.add("-an");command.add(saveMediaPath + videoResourcesPath.substring(videoResourcesPath.lastIndexOf("\\") + 1));commandStart(command);}/*** 无声视频+音频合并为一个视频* 若音频比视频长,画面停留在最后一帧,继续播放声音。** @param videoResourcesPath 视频文件的路径* @param audioResourcesPath 音频文件的路径*/public static void mergeSilent_VideoAudio(String videoResourcesPath, String audioResourcesPath) {List<String> command = new ArrayList<>();command.add(ffmpeg);command.add("-i");command.add(videoResourcesPath);command.add("-i");command.add(audioResourcesPath);command.add("-vcodec");command.add("copy");command.add("-acodec");command.add("copy");command.add(saveMediaPath + videoResourcesPath.substring(videoResourcesPath.lastIndexOf("\\") + 1));commandStart(command);}/*** 有声视频+音频合并为一个视频。* 若音频比视频长,画面停留在最后一帧,继续播放声音,* 若要以视频和音频两者时长短的为主,放开注解启用-shortest。** @param videoResourcesPath 视频文件的路径* @param audioResourcesPath 音频文件的路径*/public static void mergeVideoAudio(String videoResourcesPath, String audioResourcesPath) {List<String> command = new ArrayList<>();command.add(ffmpeg);command.add("-i");command.add(videoResourcesPath);command.add("-i");command.add(audioResourcesPath);command.add("-filter_complex");command.add("amix");command.add("-map");command.add("0:v");command.add("-map");command.add("0:a");command.add("-map");command.add("1:a");//-shortest会取视频或音频两者短的一个为准,多余部分则去除不合并//command.add("-shortest");command.add(saveMediaPath + videoResourcesPath.substring(videoResourcesPath.lastIndexOf("\\") + 1));commandStart(command);}
/*** 多视频拼接合并** @param videoResourcesPathList 视频文件路径的List*/public static void mergeVideos(List<String> videoResourcesPathList) {//时间作为合并后的视频名String getdatatime = nowTime();//所有要合并的视频转换为ts格式存到videoList里List<String> videoList = new ArrayList<>();for (String video : videoResourcesPathList) {List<String> command = new ArrayList<>();command.add(ffmpeg);command.add("-i");command.add(video);command.add("-c");command.add("copy");command.add("-bsf:v");command.add("h264_mp4toannexb");command.add("-f");command.add("mpegts");String videoTempName = video.substring(video.lastIndexOf("\\") + 1, video.lastIndexOf(".")) + ".ts";command.add(tempMediaPath + videoTempName);commandStart(command);videoList.add(tempMediaPath + videoTempName);}List<String> command1 = new ArrayList<>();command1.add(ffmpeg);command1.add("-i");StringBuffer buffer = new StringBuffer("\"concat:");for (int i = 0; i < videoList.size(); i++) {buffer.append(videoList.get(i));if (i != videoList.size() - 1) {buffer.append("|");} else {buffer.append("\"");}}command1.add(String.valueOf(buffer));command1.add("-c");command1.add("copy");command1.add(saveMediaPath + "视频合并" + getdatatime + ".mp4");commandStart(command1);}/*** 多音频拼接合并为一个音频。** @param audioResourcesPathList 音频文件路径的List*/public static void mergeAudios(List<String> audioResourcesPathList) {//时间作为合并后的音频名String getdatatime = nowTime();List<String> command = new ArrayList<>();command.add(ffmpeg);command.add("-i");StringBuffer buffer = new StringBuffer("\"concat:");for (int i = 0; i < audioResourcesPathList.size(); i++) {buffer.append(audioResourcesPathList.get(i));if (i != audioResourcesPathList.size() - 1) {buffer.append("|");} else {buffer.append("\"");}}command.add(String.valueOf(buffer));command.add("-acodec");command.add("copy");command.add(saveMediaPath + "音频合并" + getdatatime + ".mp3");commandStart(command);}/*** 视频格式转换** @param videoResourcesPath 视频文件的路径* @param format             要转换为的格式*/public static void videoFormatConversion(String videoResourcesPath, String format) {String fileName = videoResourcesPath.substring(videoResourcesPath.lastIndexOf("\\") + 1, videoResourcesPath.lastIndexOf("."));List<String> command = new ArrayList<>();command.add(ffmpeg);command.add("-i");command.add(videoResourcesPath);command.add(saveMediaPath + fileName + "." + format);commandStart(command);}/*** 获取音频或视频信息** @param videoAudioResourcesPath 音频或视频文件的路径*/public static List<String> videoAudioInfo(String videoAudioResourcesPath) {List<String> command = new ArrayList<>();command.add(ffprobe);command.add("-i");command.add(videoAudioResourcesPath);//调用命令行获取视频信息String infoStr = getInfoStr(command);System.out.println(" ");String regexDuration = "Duration: (.*?), start: (.*?), bitrate: (\\d*) kb\\/s";String regexVideo = "Video: (.*?) tbr";String regexAudio = "Audio: (.*?), (.*?) Hz, (.*?) kb\\/s";List<String> list = new ArrayList<>();Pattern pattern = Pattern.compile(regexDuration);Matcher matcher = pattern.matcher(infoStr);if (matcher.find()) {list.add("视频/音频整体信息: ");list.add("视频/音频名称:" + videoAudioResourcesPath);list.add("开始时间:" + matcher.group(2));list.add("结束时间:" + matcher.group(1));list.add("比特率: " + matcher.group(3) + " kb/s");list.add("------------------------------------ ");}Pattern patternVideo = Pattern.compile(regexVideo);Matcher matcherVideo = patternVideo.matcher(infoStr);if (matcherVideo.find()) {String videoInfo = matcherVideo.group(1);String[] sp = videoInfo.split(",");list.add("视频流信息: ");list.add("视频编码格式: " + sp[0]);int ResolutionPosition = 2;if (sp[1].contains("(") && sp[2].contains(")")) {list.add("YUV: " + sp[1] + "," + sp[2]);ResolutionPosition = 3;} else if (sp[1].contains("(") && !sp[2].contains(")") && sp[3].contains(")")) {list.add("YUV: " + sp[1] + "," + sp[2] + "," + sp[3]);ResolutionPosition = 4;} else {list.add("YUV: " + sp[1]);}list.add("分辨率: " + sp[ResolutionPosition]);list.add("视频比特率: " + sp[ResolutionPosition + 1]);list.add("帧率: " + sp[ResolutionPosition + 2]);list.add("------------------------------------ ");}Pattern patternAudio = Pattern.compile(regexAudio);Matcher matcherAudio = patternAudio.matcher(infoStr);if (matcherAudio.find()) {list.add("音频流信息: ");list.add("音频编码格式: " + matcherAudio.group(1));list.add("采样率: " + matcherAudio.group(2) + " HZ");list.add("声道: " + matcherAudio.group(3).split(",")[0]);list.add("音频比特率: " + matcherAudio.group(3).split(",")[2] + " kb/s");}return list;}/*** 视频或音频剪切* 参考@link:https://zhuanlan.zhihu.com/p/27366331** @param videoAudioResourcesPath 视频或音频文件的路径* @param startTime               开始时间* @param endTime                 结束时间*/public static void cutVideoAudio(String videoAudioResourcesPath, String startTime, String endTime) {String fileName = videoAudioResourcesPath.substring(videoAudioResourcesPath.lastIndexOf("\\") + 1);//时间作为剪切后的视频名String getdatatime = nowTime();List<String> command = new ArrayList<>();command.add(ffmpeg);command.add("-ss");command.add(startTime);command.add("-t");command.add(calculationEndTime(startTime, endTime));command.add("-i");command.add(videoAudioResourcesPath);command.add("-c:v");command.add("libx264");command.add("-c:a");command.add("aac");command.add("-strict");command.add("experimental");command.add("-b:a");command.add("98k");command.add(saveMediaPath + "剪切视频" + getdatatime + fileName);commandStart(command);}/*** 视频裁剪大小尺寸(根据leftDistance和topDistance确定裁剪的起始点,再根据finallywidth和finallyHeight确定裁剪的宽和长)* 参考@link:https://www.cnblogs.com/yongfengnice/p/7095846.html*     @link:http://www.jq-school.com/show.aspx?id=737** @param videoAudioResourcesPath 视频文件的路径* @param finallyWidth            裁剪后最终视频的宽度* @param finallyHeight           裁剪后最终视频的高度* @param leftDistance            开始裁剪的视频左边到y轴的距离(视频左下角为原点)* @param topDistance             开始裁剪的视频上边到x轴的距离(视频左下角为原点)**/public static void cropVideoSize(String videoAudioResourcesPath, String finallyWidth, String finallyHeight, String leftDistance, String topDistance) {String fileName = videoAudioResourcesPath.substring(videoAudioResourcesPath.lastIndexOf("\\") + 1);//时间作为剪切后的视频名String getdatatime = nowTime();List<String> command = new ArrayList<>();command.add(ffmpeg);command.add("-i");command.add(videoAudioResourcesPath);command.add("-vf");//获取视频信息得到原始视频长、宽List<String> list = FfmpegUtils.videoAudioInfo(videoAudioResourcesPath);String resolution = list.stream().filter(v -> v.contains("分辨率")).findFirst().get();String sp[] = resolution.split("x");String originalWidth = sp[0].substring(sp[0].indexOf(":") + 1).trim();String originalHeight = sp[1].substring(0, 4).trim();Integer cropStartWidth = Integer.parseInt(originalWidth) - Integer.parseInt(leftDistance);Integer cropStartHeight = Integer.parseInt(originalHeight) - Integer.parseInt(topDistance);command.add("crop=" + finallyWidth + ":" + finallyHeight + ":" + cropStartWidth + ":" + cropStartHeight);command.add(saveMediaPath + "裁剪视频" + getdatatime + fileName);commandStart(command);}/*** 计算两个时间的时间差** @param startime 开始时间,如:00:01:09* @param endtime  结束时间,如:00:08:27* @return 返回xx:xx:xx形式,如:00:07:18*/public static String calculationEndTime(String startime, String endtime) {LocalTime timeStart = LocalTime.parse(startime);LocalTime timeEnd = LocalTime.parse(endtime);long hour = HOURS.between(timeStart, timeEnd);long minutes = MINUTES.between(timeStart, timeEnd);long seconds = SECONDS.between(timeStart, timeEnd);minutes = minutes > 59 ? minutes % 60 : minutes;String hourStr = hour < 10 ? "0" + hour : String.valueOf(hour);String minutesStr = minutes < 10 ? "0" + minutes : String.valueOf(minutes);long getSeconds = seconds - (hour * 60 + minutes) * 60;String secondsStr = getSeconds < 10 ? "0" + getSeconds : String.valueOf(getSeconds);return hourStr + ":" + minutesStr + ":" + secondsStr;}/*** 视频截图** @param videoResourcesPath 视频文件的路径* @param screenshotTime     截图的时间,如:00:01:06*/public static void videoScreenshot(String videoResourcesPath, String screenshotTime) {//时间作为截图后的视频名String getdatatime = nowTime();List<String> command = new ArrayList<>();command.add(ffmpeg);command.add("-ss");command.add(screenshotTime);command.add("-i");command.add(videoResourcesPath);command.add("-f");command.add("image2");String fileName = videoResourcesPath.substring(videoResourcesPath.lastIndexOf("\\") + 1, videoResourcesPath.lastIndexOf("."));command.add(picturMediaPath + fileName + getdatatime + ".jpg");commandStart(command);}/*** 整个视频截图** @param videoResourcesPath 视频文件的路径* @param fps                截图的速度。1则表示每秒截一张;0.1则表示每十秒一张;10则表示每秒截十张图片*/public static void videoAllScreenshot(String videoResourcesPath, String fps) {/*//时间作为截图后的视频名String getdatatime = nowTime();*/List<String> command = new ArrayList<>();command.add(ffmpeg);command.add("-i");command.add(videoResourcesPath);command.add("-vf");command.add("fps=" + fps);String fileName = videoResourcesPath.substring(videoResourcesPath.lastIndexOf("\\") + 1, videoResourcesPath.lastIndexOf("."));command.add(picturMediaPath + fileName + "%d" + ".jpg");commandStart(command);}/*** 多图片+音频合并为视频** @param pictureResourcesPath 图片文件路径(数字编号和后缀不要)。如:D:\ffmpegMedia\pictur\101-你也不必耿耿于怀1.jpg 和D:\ffmpegMedia\pictur\101-你也不必耿耿于怀2.jpg。只需传D:\ffmpegMedia\pictur\101-你也不必耿耿于怀* @param audioResourcesPath   音频文件的路径*  @param fps   帧率,每张图片的播放时间(数值越小则每张图停留的越长)。0.5则两秒播放一张,1则一秒播放一张,10则一秒播放十张*/public static void pictureAudioMerge(String pictureResourcesPath, String audioResourcesPath,String fps) {//时间作为截图后的视频名String getdatatime = nowTime();List<String> command = new ArrayList<>();command.add(ffmpeg);command.add("-threads");command.add("2");command.add("-y");command.add("-r");//帧率command.add(fps);command.add("-i");command.add(pictureResourcesPath+"%d.jpg");command.add("-i");command.add(audioResourcesPath);command.add("-absf");command.add("aac_adtstoasc");//-shortest会取视频或音频两者短的一个为准,多余部分则去除不合并command.add("-shortest");String fileName = pictureResourcesPath.substring(pictureResourcesPath.lastIndexOf("\\") + 1);command.add(saveMediaPath +"视频合成"+ fileName + getdatatime + ".mp4");commandStart(command);}/*** 绘制音频波形图保存.jpg后缀可改为png** @param audioResourcesPath 音频文件的路径*/public static void audioWaveform(String audioResourcesPath) {List<String> command = new ArrayList<>();command.add(ffmpeg);command.add("-i");command.add(audioResourcesPath);command.add("-filter_complex");command.add("\"showwavespic=s=1280*240\"");command.add("-frames:v");command.add("1");String fileName = audioResourcesPath.substring(audioResourcesPath.lastIndexOf("\\") + 1, audioResourcesPath.lastIndexOf("."));//jpg可换为pngcommand.add(saveMediaPath + fileName + fileName + ".jpg");commandStart(command);}/*** 两个音频混缩合并为一个音频。* 音量参考:@link:https://blog.csdn.net/sinat_14826983/article/details/82975561** @param audioResourcesPath1 音频1文件路径* @param audioResourcesPath1 音频2文件路径的*/public static void mergeAudios(String audioResourcesPath1, String audioResourcesPath2) {//时间作为混缩后的音频名String getdatatime = nowTime();List<String> command = new ArrayList<>();command.add(ffmpeg);command.add("-i");command.add(audioResourcesPath1);command.add("-i");command.add(audioResourcesPath2);command.add("-filter_complex");command.add("amix=inputs=2:duration=longest");command.add(saveMediaPath + "音频混缩" + getdatatime + ".mp3");commandStart(command);}/*** 两个音频混缩合并为一个音频的不同声道(即一只耳机播放一个音频)。* 声道参考:@link:https://www.itranslater.com/qa/details/2583879740000044032** @param audioResourcesPath1 音频1文件路径* @param audioResourcesPath1 音频2文件路径的*/public static void mergeAudiosSoundtrack(String audioResourcesPath1, String audioResourcesPath2) {//时间作为混缩后的音频名String getdatatime = nowTime();List<String> command = new ArrayList<>();command.add(ffmpeg);command.add("-i");command.add(audioResourcesPath1);command.add("-i");command.add(audioResourcesPath2);command.add("-filter_complex");command.add("\"amerge=inputs=2,pan=stereo|c0<c0+c1|c1<c2+c3\"");command.add(saveMediaPath + "音频混缩" + getdatatime + ".mp3");commandStart(command);}/*** 获取当前时间,用于作为文件名*/public static String nowTime() {DateTimeFormatter f3 = DateTimeFormatter.ofPattern("yyyy-MM-dd-HH-mm-ss");LocalDate nowdata = LocalDate.now();LocalTime nowTime = LocalTime.now();return nowdata.atTime(nowTime).format(f3);}
}

测试: 


import utils.FfmpegUtils;
import java.io.IOException;
import java.util.*;public class Test {public static void main(String[] args) throws IOException {//视频播放FfmpegUtils.playVideoAudio("D:\\16-这几句真假声转换太好听了.mp4");//视频播放并指定循环次数FfmpegUtils.playVideoAudio("D:\\16-这几句真假声转换太好听了.mp4",1);//视频播放并指定宽高和循环次数FfmpegUtils.playVideoAudio("D:\\16-这几句真假声转换太好听了.mp4",400,700,1);//从视频中提取音频为mp3FfmpegUtils.getAudioFromVideo("D:\\16-这几句真假声转换太好听了.mp4");//从视频中提取视频为无声视频FfmpegUtils.getVideoFromAudio("D:\\16-这几句真假声转换太好听了.mp4");//无声视频+音频合并FfmpegUtils.mergeSilent_VideoAudio("D:\\507-#网愈云故事收藏馆.mp4","D:\\mp3\\16-这几句真假声转换太好听了.mp3");//格式转换FfmpegUtils.videoFormatConversion("D:\\507-#网愈云故事收藏馆.mp4","flv");//多视频拼接合并为一个mp4格式视频List<String> video = new ArrayList<>();video.add("D:\\16-这几句真假声转换太好听了.mp4");video.add("D:\\507-#网愈云故事收藏馆.mp4");video.add("D:\\101-你也不必耿耿于怀.mp4");FfmpegUtils.mergeVideos(video);//获取音频或视频信息List<String> list = FfmpegUtils.videoAudioInfo("D:\\ffmpegMedia\\16-这几句真假声转换太好听了.mp3");list.forEach(v -> System.out.println(v));//剪切视频或音频,startTime开始时间,endTime结束时间FfmpegUtils.cutVideoAudio("D:\\苏打绿带我走.mp4","00:00:00","00:01:05");//裁剪视频尺寸大小FfmpegUtils.cropVideoSize("D:\\张国荣.mp4", "600", "600", "720", "940");//有声视频+音频合并FfmpegUtils.mergeVideoAudio("D:\\507-#网愈云故事收藏馆.mp4","D:\\ffmpegMedia\\16-这几句真假声转换太好听了.mp3");//视频截图,screenshotTime是截图的时间FfmpegUtils.videoScreenshot("D:\\101-你也不必耿耿于怀.mp4","00:00:05");//视频完全截图,fps是截图的速度即多少秒截一张图FfmpegUtils.videoAllScreenshot("D:\\101-你也不必耿耿于怀.mp4","1");//多图片+音频合并为视频。0.5则两秒播放一张,1则一秒播放一张,10则一秒播放十张(数值越小则每张图停留的越长)FfmpegUtils.pictureAudioMerge("D:\\ffmpegMedia\\pictur\\101-你也不必耿耿于怀","D:\\ffmpegMedia\\101-你也不必耿耿于怀.mp3","0.5");//多音频拼接合并为一个mp3格式视频List<String> audio = new ArrayList<>();audio.add("D:\\ffmpegMedia\\16-这几句真假声转换太好听了.mp3");audio.add("D:\\ffmpegMedia\\忽然之间.mp3");audio.add("D:\\ffmpegMedia\\天空之城-李志.mp3");FfmpegUtils.mergeAudios(audio);//绘制音频波形图保存FfmpegUtils.audioWaveform("D:\\ffmpegMedia\\16-这几句真假声转换太好听了.mp3");//两个音频混缩合并为一个音频FfmpegUtils.mergeAudios("D:\\ffmpegMedia\\16-这几句真假声转换太好听了.mp3","D:\\ffmpegMedia\\天空之城-李志.mp3");//两个音频混缩合并为一个音频的不同声道(即一只耳机播放一个音频)。FfmpegUtils.mergeAudiosSoundtrack("D:\\ffmpegMedia\\16-这几句真假声转换太好听了.mp3","D:\\ffmpegMedia\\天空之城-李志.mp3");}
}

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

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

相关文章

生物学专业_江南大学微生物学(发酵)20002008历年考研专业课真题汇编

说明 1. 海量考研真题免费发布&#xff0c;欢迎关注公众号『守望考研』&#xff1b;2. 想获取本文对应的PDF文档以便打印使用&#xff0c;欢迎关注公众号了解领取方法&#xff1b;PS: PDF版文档清晰度更高、水印更小南开大学861微生物学1997-2001、2003-2011历年考研专业课真题…

error: ‘XXX‘ is defined but never used (no-unused-vars)报错的解决方案

错误原因 我的项目安装了eslint规范&#xff0c;ESLint 是在 ECMAScript/JavaScript 代码中识别和报告模式匹配的工具&#xff0c;它的目标是保证代码的一致性和避免错误 解决方案 一、在package.json文件内加入如下代码&#xff1a;然后保存重启项目。 "rules":…

平流式初沉池贮砂斗计算_?初沉池、二沉池的作用与区别-亨孚科技

初沉池的主要作用如下:1、去除沉淀物或浮游物&#xff0c;减轻后续处理设施的负荷。使细小的固体凝聚成大粒子&#xff0c;强化固液分离效果。3.它对胶体物质有一定的吸附和去除作用。4、初沉池在一定程度上起调节池塘的作用&#xff0c;对水质发挥一定的均质效应。5.一些废水处…

Navicat连接Oracle数据库失败,提示无效的用户名和密码(Invalid username and password)

1、Navicat是一款非常好用的数据库管理工具&#xff0c;可是一段时间没有使用&#xff0c;突然发现之前建立的Oracle连接无法打开&#xff0c;提示要输入旧密码和新密码以及确认新密码&#xff0c;在Navicat管理工具中连接之前超过180天的Oracle数据库&#xff0c;连接的时候&a…

把关与服务的关系_泉州代做投标书-电子标书值得信赖 - 泉州广告服务

此外&#xff0c;土壤资源对于人们的重要性不言而喻。为了推行土壤环境攻坚治理&#xff0c;江苏省共布设国控点位个&#xff0c;其中&#xff0c;基础点位个&#xff0c;风险点位个&#xff0c;背景点位个&#xff0c;为开展土壤污染调查奠定基础。这些地区监测站点的成功铺设…

口腔取模过程及注意事项_取模变形?教你三种方法,轻松防止取模变形!

点击查看更多精彩内容关键词&#xff1a;取模&#xff1b;适合人群&#xff1a;口腔修复科医生&#xff1b;共1497字 阅读4分钟在牙体修复中&#xff0c;一个完美的修复体是需要一个精确的模型和医生与技师之间的完美配合才能做到的。而因为模型变形出现返工的情况很多&#xf…

思科isis路由的优先级_华为 路由双点双向引入

点击上方蓝字关注我们哈喽&#xff0c;大家好&#xff01;我是艺博东 &#xff0c;是一个思科出身、专注于华为的网工&#xff1b;好了&#xff0c;话不多说&#xff0c;我们直接进入正题。双点双向重发布(OSPF、IS-IS)文章目录一、拓扑二、底层配置三、双点双向一、拓扑二、底…

python 根据名称获取pid_【Python学习笔记】76、常用第三方模块psutil

用Python来编写脚本简化日常的运维工作是Python的一个重要用途。在Linux下&#xff0c;有许多系统命令可以让我们时刻监控系统运行的状态&#xff0c;如ps&#xff0c;top&#xff0c;free等等。要获取这些系统信息&#xff0c;Python可以通过subprocess模块调用并获取结果。但…

linuxmove命令_move命令详解 dos命令 move的用法

Linux mmove命令详解 Linux mmove命令怎么用mmove为mtools工具命令&#xff0c;模拟MS-DOS的move命令&#xff0c;可在MS-DOS文件系统中移动现有的文件或目录&#xff0c;或是更改现有文件或目录的名称。 语法 mmove [源文件或目录...][目标文件或目录] 参数说明: [源文件或目录…

需要单机还是集群部署_单机、分布式、集群的区别与联系

一、单机结构一个系统业务量很小的时候所有的代码都放在一个项目中&#xff0c;然后这个项目部署在一台服务器上就好了&#xff0c;整个项目所有的服务都由这台服务器提供。这就是单机结构。单机结构的缺点是显而易见的&#xff0c;单机的处理能力毕竟是有限的&#xff0c;当你…

ygo游戏王卡组_游戏王:二线卡组脱颖而出的战将,混沌青眼卡组,仪式卡组的骄傲...

这次我们要说的卡组是在我的群里举办的比赛&#xff0c;脱颖而出的前三名卡组之一&#xff0c;这次我们进行的一场非常小规模的随机匹配比赛&#xff0c;而这次比赛的规则比较多&#xff0c;参赛的卡主也并不是很多只有16名&#xff0c;在整个游戏王系列里名不见经传的16个系列…

拖动卡顿_四招教你解决PS软件卡顿问题!

你是不是经常会遇到这样的问题&#xff1f;刚装的Ps速度很快&#xff0c;使用一段时间以后发现越来越卡&#xff0c;时不时还会死机崩溃&#xff1f;大多数人遇到这种问题都是选择重装软件&#xff0c;或者重做系统&#xff0c;耗时耗力不说&#xff0c;过不了多久又会出现同样…

python单元测试mock_python3的单元测试模块mock与性能测试模块cProfile

我们知道写完了代码需要自己跑一跑进行测试&#xff0c;一个写好的程序如果连测试都没有就上到生产环境是不敢想象的&#xff0c;这么做的人不是太自信就是太无知。传统测试无非就是自己运行一下程序查看结果&#xff0c;或者前后端服务进行联调&#xff0c;这里要说的是走正规…

极光实时监听怎么调用_源码分析 Sentinel 实时数据采集实现原理(图文并茂)

本篇将重点关注 Sentienl 实时数据收集&#xff0c;即 Sentienl 具体是如何收集调用信息&#xff0c;以此来判断是否需要触发限流或熔断。Sentienl 实时数据收集的入口类为 StatisticSlot。我们先简单来看一下 StatisticSlot 该类的注释&#xff0c;来看一下该类的整体定位。St…

awd赛题的flag是什么意思_网上说的“立flag”是什么意思?

展开全部如果你是一个经常混进b站的人的话&#xff0c;那你就会发现32313133353236313431303231363533e58685e5aeb931333363366233经常有人说出立了个flag或者是flag已立&#xff0c;那么这立flag是什么意思&#xff1f;立flag是什么梗&#xff1a;立flag指的是明确一件事情的结…

python股票网格交易法详解_股票最全“低买高卖”网格交易大法,值得收藏!

网格交易法&#xff0c;可以简单理解为在既定网格中实行“跌买涨卖”。首先需要制定一个【网格系统】&#xff0c;主要包括网格格数、网格密度、最大压力价格和最小支撑价格四大要素。当标的价格跌破一个网格密度时&#xff0c;就买入相应的仓数&#xff0c;每涨破一个网格密度…

python命令行tab补全_python命令行下按tab建补全的方法

此方法只在centos6上测试通过&#xff0c;其它系统木有测试1、在用户家目录下创建一个名字叫 .pythonstartup 的隐藏文件&#xff0c;写入如下内容&#xff1a;# python startup fileimport readlineimport rlcompleterimport atexitimport os# tab completionreadline.parse_a…

文字描边_6招迅速做出炫酷PPT字效!|10分钟干货第二期(文字描边)

不点蓝字&#xff0c;我们哪来故事&#xff1f;艺术让一切枯燥变有趣 作者 | 易焦躁星人微信号 | wyx19911003每周一篇&#xff0c;干货满满建议收藏&#xff0c;慢慢实践PPT酷炫【字效】第二弹来啦&#xff01;平时做PPT无从下手没灵感&#xff1f;根本原因还对功能实践的不够…

华为手机鸿蒙系统怎么样_华为自研操作系统“鸿蒙”已用于华为手机

重要信息有网友近日在社交网络上曝光了华为的自研操作系统&#xff0c;命名为“鸿蒙”&#xff0c;其已用于华为手机中(安全部分)。来自上海交通大学的一份PPT演示照片&#xff0c;图中显示&#xff0c;某教授领导华为操作系统团队开发了自主产权操作系统——鸿蒙。根据PPT描述…

chrome浏览器上传文件延迟_解决谷歌Chrome浏览器上传图片反应慢的办法

谷歌chrome浏览器本来非常好用&#xff0c;但是可能有的人在更新版本或重装电脑后&#xff0c;用着用着突然发现上传图片或文件的时候居然直接卡死!只能强制关闭后用ie上传&#xff0c;为此笔者也一度很苦恼。笔者在网上搜索答案后&#xff0c;并没有得到有效的帮助&#xff0c…