原文地址:JavaFx 创建快捷方式及设置开机启动 | Stars-One的杂货小窝

原本是想整个桌面启动器,需要在windows平台上实现开机启动,但我的软件都是jar文件,不是传统的exe文件,也不知道能不能设置开机启动,稍微搜集了资料研究了会,发现有思路,而且可以成功实现

本文只研究了如何在windows进行,不清楚macos和linux的情况,各位有具体的实现思路欢迎分享出来

简单说明

windows如何实现开机启动的?

在Windows系统中,设置软件开机启动并不是太难的事情,大多数工具类软件都是有提供开机启动的选项

那软件没有体用选项,就不能设置为开机启动了?答案当然是否定的

看到网络的教程,都说要去设置任务定时器,其实有种更为方便的做法,就是将软件或者快捷方式放在windows指定的文件夹即可

文件路径格式如下:C:\Users\starsone\AppData\Roaming\Microsoft\Windows\Start Menu\Programs\Startup,这个文件夹暂且称为启动文件夹

注意:如果你是将软件放在这个文件夹想要实现开机启动,需要保证你的软件是绿色版,所以更为推荐使用快捷方式的方式进行设置开机启动

实现思路

但是,刚开始我不确定jar文件是否也能直接被windows启动,于是便是拿了之前蓝奏云批量下载作为测试,由于我这是单文件,所以我直接把jar包放在启动文件夹,测试是可以的

所以,想要实现jar文件自动启动,思路就是给jar文件创建一个快捷方式,然后将此快捷方式移动到启动文件夹即可实现

难点在于如何使用java给文件创建快捷方式?

网上的资料十分少,有个方法还需要使用dll文件,我也不懂window开发,于是便放弃了

然后找的过程中,发现了有位大佬通过浏览微软官方文档,直接通过字节流方式实现创建了快捷方式,而且代码及其简单,于是稍微参考了他的源码,改造了个工具类,用来实现创建快捷方式及开机启动

考虑到原本的Java用户,工具类代码补充了Java版本的,用Java同学可以直接拷贝一份,直接使用,如果是Kotlin的,两份都可使用

吐槽下Java版写的有点繁琐,有些API都没有(如获取不带扩展名的文件名),Kotlin中直接有对应方法,不需要自己去处理实现...

Kotlin版

注:本工具类已集成在我的开源项目里了Stars-One/common-controls: TornadoFx的常用控件 controls for tornadofx

创建快捷方式使用

val lnkFile = File("D:\\project\\javafx\\lanzou-downloader\\out\\蓝奏云批量下载器3.2.lnk")
val targetFile = File("D:\\project\\javafx\\lanzou-downloader\\out\\蓝奏云批量下载器3.2.jar") ShortCutUtils.createShortCut(lnkFile,targetFile)

上述代码是将lnk文件输出在了同级目录,我们到文件夹中查看,可以发现已经生成成功了,点击也是能正常打开

设置某软件开机启动

val targetFile = File("D:\\project\\javafx\\lanzou-downloader\\out\\蓝奏云批量下载器3.2.jar")

ShortCutUtils.setAppStartup(targetFile)

这里可以看到,生成的快捷方式已经存在于启动文件夹,这样下次开机的时候就会自动启动软件了

源码

class ShortCutUtils{

    companion object{
/**
* 创建快捷方式
*
* @param lnkFile 快捷文件
* @param targetFile 源文件
*/
fun createShortCut(lnkFile: File, targetFile: File) {
if (!System.getProperties().getProperty("os.name").toUpperCase().contains("WINDOWS")) {
println("当前系统不是window系统,无法创建快捷方式!!")
return
} val targetPath = targetFile.path
if (!lnkFile.parentFile.exists()) {
lnkFile.mkdirs()
}
//原快捷方式存在,则删除
if (lnkFile.exists()) {
lnkFile.delete()
} lnkFile.appendBytes(headFile)
lnkFile.appendBytes(fileAttributes)
lnkFile.appendBytes(fixedValueOne)
lnkFile.appendBytes(targetPath.toCharArray()[0].toString().toByteArray())
lnkFile.appendBytes(fixedValueTwo)
lnkFile.appendBytes(targetPath.substring(3).toByteArray(charset("gbk")))
} /**
* 设置软件开机启动
*
* @param targetFile 源文件
*/
fun setAppStartup(targetFile: File) {
val lnkFile = File(targetFile.parentFile, "temp.lnk")
createShortCut(lnkFile, targetFile)
val startUpFile = File(startup, "${targetFile.nameWithoutExtension}.lnk")
//复制到启动文件夹,若快捷方式已存在则覆盖原来的
lnkFile.copyTo(startUpFile, true)
//删除缓存的快捷方式
lnkFile.delete()
} /**
* 设置软件开机启动
*
* @param targetFile 源文件路径
*/
fun setAppStartup(targetFilePath: String) {
setAppStartup(File(targetFilePath))
} /**
* 创建快捷方式
*
* @param lnkFilePath 快捷方式文件生成路径
* @param targetFilePath 源文件路径
*/
fun createShortCut(lnkFilePath: String, targetFilePath: String) {
createShortCut(File(lnkFilePath),File(targetFilePath))
}
/**
* 开机启动目录
*/
val startup = "${System.getProperty("user.home")}\\AppData\\Roaming\\Microsoft\\Windows\\Start Menu\\Programs\\Startup\\" /**
* 桌面目录
*/
val desktop = FileSystemView.getFileSystemView().homeDirectory.absolutePath + "\\" /**
* 文件头,固定字段
*/
private val headFile = byteArrayOf(
0x4c, 0x00, 0x00, 0x00, 0x01, 0x14, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00,
0xc0.toByte(), 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x46
) /**
* 文件头属性
*/
private val fileAttributes = byteArrayOf(
0x93.toByte(), 0x00, 0x08, 0x00, //可选文件属性
0x20, 0x00, 0x00, 0x00, //目标文件属性
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, //文件创建时间
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, //文件修改时间
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, //文件最后一次访问时间
0x00, 0x00, 0x00, 0x00, //文件长度
0x00, 0x00, 0x00, 0x00, //自定义图标个数
0x01, 0x00, 0x00, 0x00, //打开时窗口状态
0x00, 0x00, 0x00, 0x00, //热键
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 //未知
) private val fixedValueOne = byteArrayOf(
0x83.toByte(), 0x00, 0x14, 0x00, 0x1F, 0x50, 0xE0.toByte(), 0x4F, 0xD0.toByte(),
0x20, 0xEA.toByte(), 0x3A, 0x69, 0x10, 0xA2.toByte(),
0xD8.toByte(), 0x08, 0x00, 0x2B, 0x30, 0x30, 0x9D.toByte(), 0x19, 0x00, 0x2f
) /**
* 固定字段2
*/
private val fixedValueTwo = byteArrayOf(
0x3A, 0x5C, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x54, 0x00,
0x32, 0x00, 0x04, 0x00, 0x00, 0x00, 0x67, 0x50, 0x91.toByte(), 0x3C, 0x20, 0x00
) } }

Java版

创建快捷方式使用

File lnkFile = new File("D:\\project\\javafx\\lanzou-downloader\\out\\蓝奏云批量下载器3.2.lnk");
File targetFile = new File("D:\\project\\javafx\\lanzou-downloader\\out\\蓝奏云批量下载器3.2.jar"); //创建快捷方式(可以传File对象或者是路径)
ShortCutUtil.createShortCut(lnkFile,targetFile);

上述代码是将lnk文件输出在了同级目录,我们到文件夹中查看,可以发现已经生成成功了,点击也是能正常打开

设置某软件开机启动

File targetFile = new File("D:\\project\\javafx\\lanzou-downloader\\out\\蓝奏云批量下载器3.2.jar");

//设置开机启动(可以是File对象或者是路径)
ShortCutUtil.createShortCut(targetFile);

这里可以看到,生成的快捷方式已经存在于启动文件夹,这样下次开机的时候就会自动启动软件了

源码

package site.starsone;

import javax.swing.filechooser.FileSystemView;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.nio.channels.FileChannel; /**
* @author StarsOne
* @url <a href="http://stars-one.site">http://stars-one.site</a>
* @date Create in 2021/06/11 21:28
*/
public class ShortCutUtil { ;
/**
* 开机启动目录
*/
public final static String startup=System.getProperty("user.home")+
"\\AppData\\Roaming\\Microsoft\\Windows\\Start Menu\\Programs\\Startup\\";
/**
* 桌面目录
*/
public final static String desktop= FileSystemView.getFileSystemView().getHomeDirectory().getAbsolutePath()+"\\";
/**
* 文件头,固定字段
*/
private static byte[] headFile={0x4c,0x00,0x00,0x00,
0x01, 0x14,0x02,0x00,0x00,0x00,0x00,0x00,
(byte) 0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x46
};
/**
* 文件头属性
*/
private static byte[] fileAttributes={(byte) 0x93,0x00,0x08,0x00,//可选文件属性
0x20, 0x00, 0x00, 0x00,//目标文件属性
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,//文件创建时间
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,//文件修改时间
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,//文件最后一次访问时间
0x00,0x00,0x00,0x00,//文件长度
0x00,0x00,0x00,0x00,//自定义图标个数
0x01,0x00,0x00,0x00,//打开时窗口状态
0x00,0x00,0x00,0x00,//热键
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00//未知
};
/**
* 固定字段1
*/
static byte[] fixedValueOne={
(byte) 0x83 ,0x00 ,0x14 ,0x00
,0x1F ,0x50 ,(byte)0xE0 ,0x4F
,(byte)0xD0 ,0x20 ,(byte)0xEA
,0x3A ,0x69 ,0x10 ,(byte)0xA2
,(byte)0xD8 ,0x08 ,0x00 ,0x2B
,0x30,0x30,(byte)0x9D,0x19,0x00,0x2f
};
/**
* 固定字段2
*/
static byte[] fixedValueTwo={
0x3A ,0x5C ,0x00 ,0x00 ,0x00 ,0x00 ,0x00 ,0x00 ,0x00 ,0x00
,0x00 ,0x00 ,0x00 ,0x00 ,0x00 ,0x00 ,0x00 ,0x00 ,0x00 ,0x00
,0x00 ,0x54 ,0x00 ,0x32 ,0x00 ,0x04
,0x00 ,0x00 ,0x00 ,0x67 ,0x50 ,(byte)0x91 ,0x3C ,0x20 ,0x00
}; /**
* 生成快捷方式
* @param start 完整的文件路径
* @param target 完整的快捷方式路径
*/
private static void start(String start,String target){
FileOutputStream fos= null;
try {
fos = new FileOutputStream(createDirectory(start));
fos.write(headFile);
fos.write(fileAttributes);
fos.write(fixedValueOne);
fos.write((target.toCharArray()[0]+"").getBytes());
fos.write(fixedValueTwo);
fos.write(target.substring(3).getBytes("gbk"));
fos.flush();
} catch (Exception e) {
e.printStackTrace();
}finally {
if(fos!=null) {
try {
fos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
} /**
* 解决父路径问题
*/
private static File createDirectory(String file){
File f=new File(file);
//获取父路径
File fileParent = f.getParentFile();
//如果文件夹不存在
if (fileParent!=null&&!fileParent.exists()) {
//创建文件夹
fileParent.mkdirs();
}
if (f.exists()) {
f.delete();
} return f;
} /**
* 复制文件
* @param source
* @param dest
* @throws IOException
*/
private static void copyFileUsingFileChannels(File source, File dest) throws IOException {
FileChannel inputChannel = null;
FileChannel outputChannel = null;
try {
inputChannel = new FileInputStream(source).getChannel();
outputChannel = new FileOutputStream(dest).getChannel();
outputChannel.transferFrom(inputChannel, 0, inputChannel.size());
} finally {
inputChannel.close();
outputChannel.close();
}
} /**
* 创建快捷方式
* @param lnkFilePath 快捷方式文件路径
* @param targetFilePath 快捷方式对应源文件的文件路径
*/
public static void createShortCut(String lnkFilePath,String targetFilePath) {
if (!System.getProperties().getProperty("os.name").toUpperCase().contains("WINDOWS")) {
System.out.println("当前系统不是window系统,无法创建快捷方式!!");
return;
}
start(lnkFilePath,targetFilePath);
} /**
* 生成快捷方式
* @param lnkFile 快捷方式文件
* @param targetFile 快捷方式对应源文件
*/
public static void createShortCut(File lnkFile,File targetFile) { if (!System.getProperties().getProperty("os.name").toUpperCase().contains("WINDOWS")) {
System.out.println("当前系统不是window系统,无法创建快捷方式!!");
return;
}
start(lnkFile.getPath(),targetFile.getPath());
} /**
* 设置某软件开启启动
* @param targetFile 源文件
* @return 是否设置成功
*/
public static boolean setAppStartup(File targetFile) { File lnkFile = new File(targetFile.getParent(),"temp.lnk");
createShortCut(lnkFile,targetFile);
try {
//将软件复制到软件想
copyFileUsingFileChannels(lnkFile, new File(startup,lnkFile.getName()));
return true;
} catch (IOException e) {
System.out.println("移动到startup文件夹失败");
return false;
}
} /**
* 设置某软件开启启动
* @param targetFilePath 源文件路径
* @return 是否设置成功
*/
public static boolean setAppStartup(String targetFilePath) {
File targetFile = new File(targetFilePath);
return setAppStartup(targetFile);
}
}

参考

JavaFx 创建快捷方式及设置开机启动的更多相关文章

  1. windows设置开机启动项

    一.windows下设置开机启动有如下方法 1 注册表启动项目RUN 2 计划任务,在"windows管理">"计划任务管理器"中新建任务,在操作栏指定要 ...

  2. openerp 7 在ubuntu上设置开机启动

    我们要让openerp开机运行起来. 第一步,先进入系统目录: cd /etc/init.d 第二步,创建文件.命名为openerp-server sudo vi openepr-server 第三步 ...

  3. windows下架设SVN服务器并设置开机启动

    原文:windows下架设SVN服务器并设置开机启动 1.安装SVN服务器,到http://subversion.apache.org/packages.html上下载windows版的SVN,并安装 ...

  4. ubuntu-18.04 设置开机启动脚本-亲测有效

    ubuntu-18.04不能像ubuntu14一样通过编辑rc.local来设置开机启动脚本,通过下列简单设置后,可以使rc.local重新发挥作用. 2.将下列内容复制进rc-local.servi ...

  5. ubuntu-18.04 设置开机启动脚本

    ubuntu-18.04 设置开机启动脚本 参阅下列链接 https://askubuntu.com/questions/886620/how-can-i-execute-command-on-sta ...

  6. Ubuntu 18.04 启用 rc.local 设置开机启动

    ubuntu18.04 不再使用initd管理系统,改用systemd. 然而systemd很难用,改变太大,跟之前的完全不同. 使用systemd设置开机启动为了像以前一样,在/etc/rc.loc ...

  7. Ubuntu 18.04 设置开机启动脚本 rc.local systemd

    ubuntu18.04不再使用initd管理系统,改用systemd. ubuntu-18.04不能像ubuntu14一样通过编辑rc.local来设置开机启动脚本,通过下列简单设置后,可以使rc.l ...

  8. ubuntu高版本如何设置开机启动脚本

    ubuntu-18.04不能像ubuntu14一样通过编辑rc.local来设置开机启动脚本 可以通过下列简单设置后,可以使rc.local重新发挥作用. 1.建立rc-local.service文件 ...

  9. Linux上设置开机启动Java程序

    在Linux上设置开机启动Java程序,例如:test.jar 在Linux上启动Java程序的命令: nohup java -jar test.jar >/dev/>& & ...

随机推荐

  1. 报错com.github.pagehelper.PageHelper cannot be cast to com.github.pagehelper.Dialect

    报错com.github.pagehelper.PageHelper cannot be cast to com.github.pagehelper.Dialect spring以及mybatis版本 ...

  2. 建立AD域,修改密码后不重启生效命令

    net user administrator /passwordreq:yes

  3. 基于texlive定制chemfig化学式转换Python服务镜像

    chemfig 据别人介绍,在绘制平面分子式,乃至化学反应式.机理图时,大家使用的基本都是ChemDraw.当然ChemDraw是一款强大的软件,无论是平面的还是立体的分子结构式都能毫不费力地绘制出来 ...

  4. 教你用SQL实现统计排名

    前言: 在某些应用场景中,我们经常会遇到一些排名的问题,比如按成绩或年龄排名.排名也有多种排名方式,如直接排名.分组排名,排名有间隔或排名无间隔等等,这篇文章将总结几种MySQL中常见的排名问题. 创 ...

  5. 纯vue3实现的svg可视化web组态编辑器。主要用于物联网mqtt实时系统图

    vue就是边做这个项目边学的 代码可能有点乱 还望各位大神勿喷 如果代码对您有帮助 麻烦辛苦帮我点个star 预览地址 https://svg.yaolunmao.top 如何使用 # 克隆项目 gi ...

  6. 发布声明$\beta$

    一.新功能 \(\beta\)阶段集中开发了3大核心功能:支持模块的嵌套.模型市场.模型推理,这三项基本上都是从零开始.徒手开发的功能,没有轮子可以参照,因此也不具有可以对比的先前版本. 除此之外,开 ...

  7. CAS的理解

    CAS(CompareAndSweep)工作方式 ​ CAS是乐观锁技术,当多个线程尝试使用CAS同时更新同一个变量时,只有其中一个线程能更新变量的值,而其它线程都失败,失败的线程并不会被挂起,而是被 ...

  8. [Java] 类库例题

    例1 字符串操作 定义一个StringBuffer类对象,然后通过append()方法向对象中添加26个小写字母,每次只添加一次,共添加26次,然后按逆序方式输出,并且可以删除前5个字符 面向过程实现 ...

  9. 阿里云上安装 OpenStack 是什么体验

    阿里云上跑火车(安装 OpenStack Train 版本),猜猜最终花了多少钱? 前言 前面给大家提供了用虚拟机安装 OpenStack 的镜像,虽然已经很简便了,但还是略显笨重.一来镜像文件比较大 ...

  10. Linux 系统定时任务:crontab,anacron

    Linux 系统定时任务:crontab,anacron 一.Cron 服务 1. 启动服务 service cron start 2. 关闭服务 service cron stop 3. 重启服务 ...