javaFX裁剪视频exe

本文档介绍了一个使用JavaFX和ffmpeg的程序,实现视频裁剪功能,支持自定义格式输入,并通过线程并发处理进度显示,确保界面流畅。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

该程序需要用到ffmpeg
链接:https://pan.baidu.com/s/1GEc6CwC7_QEF44SHw6X5hA
提取码:w4do

该项目就不发了
只发主要代码

package sample;

import javafx.concurrent.Service;
import javafx.concurrent.Task;
import javafx.concurrent.WorkerStateEvent;
import javafx.event.ActionEvent;
import javafx.fxml.FXML;
import javafx.fxml.Initializable;
import javafx.scene.control.Button;
import javafx.scene.control.ProgressBar;
import javafx.scene.control.TextField;
import javafx.scene.text.Text;
import javafx.stage.DirectoryChooser;
import javafx.stage.Stage;

import java.io.BufferedReader;
import java.io.File;
import java.io.InputStreamReader;
import java.net.URL;
import java.util.ResourceBundle;

/*
 * author:L
 * time:2021/12/22
 * context:新加功能自定义格式的输入*/

/*
 * author:L
 * time:2021/12/23
 * context:将for循环改成了递归处理
 *  好处:不用再规定几层文件夹了
 * */

/*
 * author:L
 * time:2021/12/27
 * context:将裁剪视频功能和进度功能单独做成两个线程,排除在主线程之外
 *  好处:若用主线程裁剪视频会导致主线程一直运行裁剪,从而导致界面卡住。
 *      虽然进度单开了一个线程且线程内的数据在更新,但是界面卡住了,进度
 *      线程内更新的数据无法显示。现在裁剪视频也单独用一个线程而非是主线
 *      程,这样界面不会卡住,进度线程更新的数据就能实时在界面上显示了。
 * */

public class Controller implements Initializable{

    public static int jinduNum;
    static boolean isEnd = false;
    String getfilepath;
    String zidingyiCanShu = "100:100";
    @FXML
    private TextField XText;//自定义x坐标点输入框
    @FXML
    private TextField YText;//自定义y坐标点输入框
    @FXML
    private ProgressBar jindu;//进度条
    @FXML
    private Button bth2;
    @FXML
    private Text process3;
    @FXML
    private Button But1;

    @Override
    public void initialize(URL location, ResourceBundle resources) {
        bth2.setOnAction(e->getUpdateFliePathDemo(e));//按钮注册事件处理函数
    }

    /*开始计数按钮对应方法
    	用于提示进度
    */
    
    public void getUpdateFliePathDemo(ActionEvent e){

        Service<String> service=new Service<String>() {
            @Override
            protected Task<String> createTask() {
                return new Task<String>() {
                    @Override
                    protected String call() throws Exception {
                        for (int a = 1; a <= 1000000000; a++) {
                            if(isEnd){
                                break;
                            }
//                            //更新service的value属性
                            updateValue("正在处理第 " + jinduNum + " 个");
                            Thread.sleep(10000);

                        }
                        return "完成:一共处理 "+jinduNum+" 个视频";
                    }
                };
            }
        };
        //绑定process3的text属性为service的text属性
        process3.textProperty().bind(service.valueProperty());
        //任务完成时会调用
        service.setOnSucceeded((WorkerStateEvent event) -> {
            System.out.println("任务处理完成!");
        });
        //启动任务start()一定是最后才调用的
        service.start();
    }

    //目录按钮对应的方法
    public void getFilePathDemo() {
        DirectoryChooser directoryChooser = new DirectoryChooser();
        //默认打开的路径
        directoryChooser.setInitialDirectory(new File("C:\\"));
        Stage stage = new Stage();
        //页面传入的路径
        this.getfilepath = directoryChooser.showDialog(stage).toString();
        File file = new File(this.getfilepath);

        //获取自定义的格式
        String x1 = this.XText.getText();
        String y1 = this.YText.getText();
        if (!"".equals(x1) && !"".equals(y1)) {
            this.zidingyiCanShu = x1 + ":" + y1;
        }

		//进行裁剪视频操作
        new Thread(){
            @Override
            public void run() {
                super.run();
                getFilePath(file);
                isEnd = true;
            }
            //递归搜寻可见光,红外,自定义文件夹
            public void getFilePath(File file){
                if(file.isDirectory()){
                    if(file.getName().equals("可见光")){
                        videoTypeCut(file.getPath(),file.listFiles(),"可见光","kejianguang");
                    }else if(file.getName().equals("红外")){
                        videoTypeCut(file.getPath(),file.listFiles(),"红外","hongwai");
                    }else if(file.getName().equals("自定义")){
                        videoTypeCut(file.getPath(),file.listFiles(),"自定义","zidingyi");
                    }else{
                        File[] files = file.listFiles();
                        if(files.length>0){
                            for(int i=0;i<files.length;i++){
                                getFilePath(files[i]);
                            }
                        }
                    }
                }
            }

            //判断保存文件夹是否有,调用裁剪视频代码
            public void videoTypeCut(String thisPath,File[] visibleLightPaths,String typeName,String type){
                //判断裁剪可见光文件夹有没有,没有就创建
                String newPath = thisPath.replaceAll(typeName,"剪切"+typeName);
                File caiJianFile = new File(newPath);
                if(!caiJianFile.exists()){
                    caiJianFile.mkdir();
                }
                for(int i=0;i<visibleLightPaths.length;i++){
                    String thisFileName = visibleLightPaths[i].getName();
                    try {
                        splitVideo(thisPath+"\\"+thisFileName, newPath+"\\", thisFileName, type);
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
            }

            //执行按钮对应方法
            public void splitVideo(String finalPath, String outPath, String fileName, String type) throws Exception {
                jinduNum++;
                System.out.println(finalPath + "  裁剪开始");
                System.out.println("正在裁剪,请勿操作或中断");
                String cmd = "";
                if (type == "kejianguang") {
                    cmd = "ffmpeg -i " + finalPath + " -vf crop=1920:960:0:0 " + outPath + fileName + " -y";
                } else if (type == "hongwai") {
                    cmd = "ffmpeg -i " + finalPath + " -vf crop=720:520:0:0 " + outPath + fileName + " -y";
                } else if (type == "zidingyi") {
                    cmd = "ffmpeg -i " + finalPath + " -vf crop=" + zidingyiCanShu + ":0:0 " + outPath + fileName + " -y";
                }
                Process process = Runtime.getRuntime().exec(cmd);
                BufferedReader br = new BufferedReader(new InputStreamReader(process.getErrorStream()));
                String line;
                while ((line = br.readLine()) != null) {
//            System.out.println(line);
                }
                boolean btnSet = true;
                process.waitFor();

                System.out.println(finalPath + "  裁剪结束");
                System.out.println();
            }
        }.start();
        System.out.println("程序结束,请查看结果");
    }


}


<?xml version="1.0" encoding="UTF-8"?>

<?import java.lang.*?>
<?import javafx.scene.text.*?>
<?import javafx.scene.control.*?>
<?import javafx.scene.layout.*?>

<AnchorPane maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="400.0" prefWidth="600.0" stylesheets="/sample/sty.css" xmlns="http://javafx.com/javafx/8" xmlns:fx="http://javafx.com/fxml/1" fx:controller="sample.Controller">
   <children>
      <Label layoutX="120.0" layoutY="43.0" text="视频分辨率裁剪工具">
         <font>
            <Font size="24.0" />
         </font>
      </Label>

      <Label layoutX="110.0" layoutY="193.0" text="目标目录:" />
      <Button fx:id="But1" layoutX="178.0" layoutY="186.0" mnemonicParsing="false" onAction="#getFilePathDemo" text="选择文件夹">
         <font>
            <Font size="14.0" />
         </font>
      </Button>

      <Label layoutX="108.0" layoutY="135.0" text="自定义格式:" />
      <Label layoutX="175.0" layoutY="135.0" text="x" />
      <TextField fx:id="XText" layoutX="187.0" layoutY="131.0" prefHeight="23.0" prefWidth="72.0" />
      <Label layoutX="264.0" layoutY="135.0" text="y" />
      <TextField fx:id="YText" layoutX="277.0" layoutY="131.0" prefHeight="23.0" prefWidth="72.0" />

      <Button fx:id="bth2" layoutX="283.0" layoutY="189.0" mnemonicParsing="false" text="开始计数" />

      <Label layoutX="43.0" layoutY="304.0" text="提示:" textFill="#fc0000">
         <font>
            <Font size="18.0" />
         </font>
      </Label>
      <Label layoutX="94.0" layoutY="309.0" text="如有自定义格式,请先选择填写自定义格式,然后选择目标目录。" />
      <Label layoutX="94.0" layoutY="328.0" text="选择好文件夹,点击计数后就不要动了,程序已经在裁剪视频了。" />
<!--      <ProgressBar fx:id="jindu" layoutX="201.0" layoutY="191.0" prefWidth="200.0" progress="0" />-->
      <Text fx:id="process3" layoutX="114.0" layoutY="268.0" strokeType="OUTSIDE" strokeWidth="0.0" text="当前未开始裁剪" wrappingWidth="401.6513671875" />
      <Label layoutX="151.0" layoutY="171.0" />

<!--      <Button fx:id="bth3" layoutX="244.0" layoutY="237.0" mnemonicParsing="false" onAction="#splitVideo" text="准备就绪开始执行" />-->
   </children>
</AnchorPane>

<think>嗯,用户现在问的是C++ Qt和JavaFX开发的桌面软件在使用上的区别。首先,我需要确认用户的具体需求。他们可能是在选择桌面开发框架,所以想知道这两种技术的优缺点,以便做出决策。 先回想之前用户的历史问题,他们之前询问过深度学习相关的内容,现在转向桌面开发,可能是在项目开发中需要选择合适的技术栈。用户身份可能是开发者或者学生,想要了解不同框架的特点。 接下来,我需要比较Qt和JavaFX的主要区别。首先,编程语言方面,Qt用的是C++,而JavaFX自然用Java。C++的性能通常更高,适合对性能要求高的应用,比如图形密集型软件。JavaFX虽然性能也不错,但Java的垃圾回收和虚拟机可能会带来一些延迟,不过对于一般桌面应用足够。 然后是跨平台支持。Qt以原生跨平台著称,能编译成各平台的原生代码,界面风格也会适配系统。JavaFX依赖JVM,虽然也是跨平台,但需要用户安装JRE,可能影响部署体验。 界面设计工具方面,Qt有强大的Qt Designer,支持拖拽设计,而JavaFX使用FXML和Scene Builder,两者功能类似,但Qt的历史更久,工具链更成熟。 许可证和商业支持也是重点。Qt的商业版需要付费,适合企业级应用,而社区版有LGPL协议。JavaFX是开源的,完全免费,适合预算有限的项目。 生态系统和社区支持方面,Qt有成熟的生态,涵盖更多领域,比如嵌入式。JavaFX作为Java的一部分,有丰富的库和框架,但桌面开发方面资源可能不如Qt多。 内存和资源管理方面,C++需要手动管理,但效率高;Java自动垃圾回收,简化开发但可能有性能开销。 线程和并发处理,Qt的信号槽机制简化了线程通信,而JavaFX的Platform.runLater需要手动处理UI更新,稍显繁琐。 部署方面,Qt打包成原生二进制,JavaFX需要JRE,用户可能需要额外安装,或者用jlink创建自定义运行时。 最后,应用场景。Qt适合高性能、跨平台原生应用,尤其是工业、嵌入式领域。JavaFX适合需要快速开发、跨平台且已有Java经验的团队。 用户可能关心开发效率、性能、部署难易和成本。比如,如果用户的项目需要高性能,或者涉及嵌入式系统,Qt更合适。如果是需要快速开发,团队熟悉Java,那么JavaFX更好。需要综合考虑这些因素来回答。</think>以下是关于使用 **C++ Qt** 和 **JavaFX** 开发桌面软件的主要区别分析,从 **性能、跨平台能力、开发效率、部署方式** 等多个维度展开: --- ### **1. 核心语言与运行环境** | **特性** | **C++ Qt** | **JavaFX** | |-------------------|---------------------------------------------|--------------------------------------------| | **编程语言** | C++(支持部分C++17特性) | Java(需JDK支持,通常结合Java 8+) | | **运行依赖** | 编译为原生二进制,无额外运行时 | 依赖JVM(需安装JRE/JDK) | | **性能特点** | 接近硬件底层,内存管理高效,适合高性能场景 | 受JVM垃圾回收机制影响,响应速度略低 | **示例场景**: - 开发视频编辑软件时,Qt的C++原生性能在处理高分辨率视频渲染时更具优势。 - JavaFX开发数据管理工具时,JVM的自动内存管理简化了开发复杂度。 --- ### **2. 跨平台能力对比** | **维度** | **Qt** | **JavaFX** | |-------------------|---------------------------------------------|--------------------------------------------| | **编译方式** | 源码级跨平台(需重新编译为各平台二进制) | "Write Once, Run Anywhere"(字节码跨平台) | | **界面一致性** | 默认使用系统原生控件(可定制为统一风格) | 通过CSS自定义样式,但控件渲染依赖JVM | | **移动端支持** | 支持iOS/Android(Qt for Mobile) | 仅限桌面端(需第三方框架支持移动端) | **典型问题**: - Qt在Linux嵌入式设备(如工控机)上可直接编译部署,而JavaFX需确保目标设备有兼容的JVM版本。 --- ### **3. 开发工具与生态** | **工具链** | **Qt** | **JavaFX** | |-------------------|---------------------------------------------|--------------------------------------------| | **IDE支持** | Qt Creator(专用IDE)或VS Code等 | IntelliJ IDEA、Eclipse、NetBeans | | **界面设计器** | Qt Designer(拖拽式UI设计,生成`.ui`文件) | Scene Builder(生成FXML文件) | | **第三方库生态** | 通过Qt Modules提供数据库、网络、3D渲染等 | 依赖Java生态(如Spring、Hibernate) | **代码示例对比**: - **Qt按钮点击事件**(C++): ```cpp QObject::connect(button, &QPushButton::clicked, []() { qDebug() << "Button clicked!"; }); ``` - **JavaFX按钮点击事件**(Java): ```java button.setOnAction(event -> { System.out.println("Button clicked!"); }); ``` --- ### **4. 部署与安装包** | **部署方式** | **Qt** | **JavaFX** | |-------------------|---------------------------------------------|--------------------------------------------| | **打包体积** | 较小(仅包含必要动态库) | 较大(需捆绑JRE或使用jlink裁剪运行时) | | **安装复杂度** | 用户无需额外操作(原生应用) | 需引导用户安装JRE(或内嵌JRE) | | **商业授权** | 需购买商业授权(若闭源分发) | 完全免费(Oracle JDK需注意许可条款) | **案例说明**: - 使用Qt开发的工业控制软件通常直接打包为`.exe`/`.dmg`,用户双击即可运行。 - JavaFX应用若未内嵌JRE,用户首次打开时可能需手动安装Java环境。 --- ### **5. 适用场景推荐** - **优先选择Qt**: ✅ 需要高性能计算(如CAD、游戏引擎) ✅ 嵌入式设备开发(如医疗设备、车载系统) ✅ 对安装包体积敏感(如轻量级工具软件) - **优先选择JavaFX**: ✅ 已有Java技术栈团队(快速上手) ✅ 需要与Java后端深度集成(如企业级管理系统) ✅ 对跨平台一致性要求高(不依赖原生控件) --- ### **6. 趋势与扩展性** - **Qt**:向3D(Qt 3D Studio)、物联网(Qt for MCU)延伸,商业支持完善。 - **JavaFX**:Oracle已将其捐献给开源社区(OpenJFX),依赖社区维护,未来更多集成于Java生态工具链。 --- **总结选择建议**: - 若追求**极致性能**和**原生体验**,选**Qt**; - 若需要**快速开发**且团队熟悉Java,选**JavaFX**。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值