Ebben a java alkalmazásban egy videót próbálok kis klippekké konvertálni.

Itt van a megvalósítási osztály ugyanarra

package ffmpeg.clip.process; import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.nio.file.Paths; import java.time.Duration; import java.time.LocalTime; import java.time.temporal.ChronoUnit; import java.util.concurrent.CompletableFuture; import java.util.concurrent.ExecutionException; import java.util.stream.Collectors; import ffmpeg.clip.utils.VideoConstant; import ffmpeg.clip.utils.VideoUtils; /* * @author Nitishkumar Singh * @Description: class will use ffmpeg to break an source video into clips */ public class VideoToClip { /* * Prevent from creating instance */ private VideoToClip() { } /** * Get Video Duration is milliseconds * * @Exception IOException - File does not exist VideoException- Video File have data issues */ static LocalTime getDuration(String sourceVideoFile) throws Exception { if (!Paths.get(sourceVideoFile).toFile().exists()) throw new Exception("File does not exist!!"); Process proc = new ProcessBuilder(VideoConstant.SHELL, VideoConstant.SHELL_COMMAND_STRING_ARGUMENT, String.format(VideoConstant.DURATION_COMMAND, sourceVideoFile)).start(); boolean errorOccured = (new BufferedReader(new InputStreamReader(proc.getErrorStream())).lines() .count() > VideoConstant.ZERO); String durationInSeconds = new BufferedReader(new InputStreamReader(proc.getInputStream())).lines() .collect(Collectors.joining(System.lineSeparator())); proc.destroy(); if (errorOccured || (durationInSeconds.length() == VideoConstant.ZERO)) throw new Exception("Video File have some issues!"); else return VideoUtils.parseHourMinuteSecondMillisecondFormat(durationInSeconds); } /** * Create Clips for Video Using Start and End Second * * @Exception IOException - Clip Creation Process Failed InterruptedException - Clip Creation task get"s failed */ static String toClipProcess(String sourceVideo, String outputDirectory, LocalTime start, LocalTime end, String fileExtension) throws IOException, InterruptedException, ExecutionException { String clipName = String.format(VideoConstant.CLIP_FILE_NAME, VideoUtils.getHourMinuteSecondMillisecondFormat(start), VideoUtils.getHourMinuteSecondMillisecondFormat(end), fileExtension); String command = String.format(VideoConstant.FFMPEG_OUTPUT_COMMAND, sourceVideo, VideoUtils.getHourMinuteSecondMillisecondFormat(start), VideoUtils.getHourMinuteSecondMillisecondFormat(end.minus(start.toNanoOfDay(), ChronoUnit.NANOS)), outputDirectory, clipName); LocalTime startTime = LocalTime.now(); System.out.println("Clip Name: " + clipName); System.out.println("FFMPEG Process Execution Started"); CompletableFuture<Process> completableFuture = CompletableFuture.supplyAsync(() -> { try { return executeProcess(command); } catch (InterruptedException | IOException ex) { throw new RuntimeException(ex); } }); completableFuture.get(); // remove LocalTime endTime = LocalTime.now(); System.out.println("Clip Name: " + clipName); System.out.println("FFMPEG Process Execution Finished"); System.out.println("Duration: " + Duration.between(startTime, endTime).toMillis() / 1000); return clipName; } /** * Create and Execute Process for each command */ static Process executeProcess(String command) throws InterruptedException, IOException { Process clipProcess = Runtime.getRuntime().exec(command); clipProcess.waitFor(); return clipProcess; } } 

A teljes megoldás elérhető a Github címen. Valójában a következőt használom: CompletableFuture , és az FFMPEG parancsot futtatom a Java folyamat létrehozásával. Túl sok az idő, amire szükség van. Egy 40 perces videó elkészítése több mint 49 percet vesz igénybe egy 64-es CPU-gépen. Megpróbálom csökkenteni a mag méretét 8-ra vagy valamire, valamint javítani a teljesítményét, mivel ez a fajta teljesítmény nem lesz elfogadható semmiféle alkalmazáshoz.

2017. január 22-i frissítés
Egy frissítés, megváltoztattam az FFMPEG parancsot klipek létrehozására és FFMPEG 3-ra frissítettem, de nincs javulás.

ffmpeg -y -i INPUT_FILE_PATH -ss TIME_STAMP -t DURATION_TO_CLIP OUTPUT_FILE_PATH

Megjegyzések

  • Néhány példát közzé tettem a JavaCV segítségével, nem ‘ próbáltam hosszú video-videóval, de rövid videókhoz (pl. 5 perc) a feldolgozási idő megközelítőleg 1 perc.

Válasz

Ez a videó természetes korlátozása kódolás. A modern gépeken 1 perc 720p videó körülbelül 1 perc alatt kódolódik.

Sok időt spórolhat meg, ha nincs szüksége újrakódolásra (azaz. kodek vagy videó méretének megváltoztatása) a -codec copy ffmpeg opció használatával.

Azt is mondtad, hogy 64 magod van, de a kódod csak 1 szálat használ a kódoláshoz. Használja a -threads 0 parancsot, hogy az ffmpeg maga válasszon.

Ha ezt Java-ban kell végrehajtania, adja meg az Jaffree egy esély (szerző vagyok).

Hozzászólások

  • Ha Jaffree-t írt, kérem, mondja meg ezt kifejezetten .
  • github.com/bramp/ffmpeg-cli-wrapper jobb megoldás lehet az Ön számára.
  • Pontosan az ffmpeg-cli-wrapperrel kapcsolatos számos probléma miatt indítottam el a Jaffree-t.

Válasz

Tudom, hogy ez egy régi kérdés, de úgy gondolom, hogy ez hasznos lehet a Java fejlesztők számára.

Van egy szép JavaCV könyvtár, ez az live több C és C ++ könyvtár, például az FFmpeg burkolója.

Ez egy egyszerű példa az átalakító megvalósítására:

import org.bytedeco.javacpp.avcodec; import java.text.DateFormat; import java.text.SimpleDateFormat; import java.util.Date; public class PacketRecorderTest { private static final DateFormat DATE_FORMAT = new SimpleDateFormat("yyyyMMdd__hhmmSSS"); private static final int RECORD_LENGTH = 5000; private static final boolean AUDIO_ENABLED = false; public static void main(String[] args) throws FrameRecorder.Exception, FrameGrabber.Exception { String inputFile = "/home/usr/videos/VIDEO_FILE_NAME.mp4"; // Decodes-encodes String outputFile = "/tmp/" + DATE_FORMAT.format(new Date()) + "_frameRecord.mp4"; PacketRecorderTest.frameRecord(inputFile, outputFile); // copies codec (no need to re-encode) outputFile = "/tmp/" + DATE_FORMAT.format(new Date()) + "_packetRecord.mp4"; PacketRecorderTest.packetRecord(inputFile, outputFile); } public static void frameRecord(String inputFile, String outputFile) throws FrameGrabber.Exception, FrameRecorder.Exception { int audioChannel = AUDIO_ENABLED ? 1 : 0; FFmpegFrameGrabber grabber = new FFmpegFrameGrabber(inputFile); FFmpegFrameRecorder recorder = new FFmpegFrameRecorder(outputFile, 1280, 720, audioChannel); grabber.start(); recorder.start(); Frame frame; long t1 = System.currentTimeMillis(); while ((frame = grabber.grabFrame(AUDIO_ENABLED, true, true, false)) != null) { recorder.record(frame); if ((System.currentTimeMillis() - t1) > RECORD_LENGTH) { break; } } recorder.stop(); grabber.stop(); } public static void packetRecord(String inputFile, String outputFile) throws FrameGrabber.Exception, FrameRecorder.Exception { int audioChannel = AUDIO_ENABLED ? 1 : 0; FFmpegFrameGrabber grabber = new FFmpegFrameGrabber(inputFile); FFmpegFrameRecorder recorder = new FFmpegFrameRecorder(outputFile, 1280, 720, audioChannel); grabber.start(); recorder.start(grabber.getFormatContext()); avcodec.AVPacket packet; long t1 = System.currentTimeMillis(); while ((packet = grabber.grabPacket()) != null) { recorder.recordPacket(packet); if ((System.currentTimeMillis() - t1) > RECORD_LENGTH) { break; } } recorder.stop(); grabber.stop(); } } 

Ez az alapvető megvalósítás bemutatja, hogyan lehet videót konvertálni egy FFmpeg csomag vagy egy JavaCV keret .

Megjegyzések

  • Üdvözöljük a Code Review webhelyen. Ön bemutatott egy alternatív megoldást, de még ‘ nem ellenőrizte a kódot. Kérjük, magyarázza el az érvelését (hogyan működik a megoldása, és miért jobb az eredetinél), hogy a szerző és más olvasók tanulhassanak az ön gondolkodási folyamatából.

Vélemény, hozzászólás?

Az email címet nem tesszük közzé. A kötelező mezőket * karakterrel jelöltük