原文地址: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. Python中pip安装报错Unable to create process using '....'

    因为我本人在电脑上安装了python2和python3,所以在安装的时候,把两个python的安装目录都安装到G盘了.然后两个python的执行文件分别改成了 python2.exe 和  pytho ...

  2. 4.PHP正则表达式与数组

    PHP正则表达式相关 行定位符 开头 ^tm 结尾 tm$ 不限制 tm 单词定界符 \btm\b   单词tm,如果想取反的话就是大写的 \Btm\B 或的关系,[Tt][Mm] 可以表达 tm T ...

  3. 【mybatis】mybatis分页拦截器搭配bootstrap-table使用

    提前说明: 这一种方式已被我自己pass掉了,已经被新的方式迭代了.但是记录下自己曾经的成果还是有必要的,而且里面的思想还是不变的,另外技术不就是在不断地迭代中升级吗.千万不要想着一步完美,那样会让你 ...

  4. 数据人必读!玩转数据可视化用这个就够了——高德LOCA API 2.0升级来袭!

    引言 "一图胜千言",大数据时代来临,数据与人们生活密切相关.复杂难懂且体量庞大的数据给人的感觉总是冷冰冰的,让人难以获取到重点信息,也找不出规律和特征,数据价值发挥不出来.空间数 ...

  5. 分布式ID

    需求 全局唯一 高性能 高可用 简单易用 UUID 优点: 唯一 不依赖于任何第三方服务 缺点: 是字符串类型而非数字,不满足数字ID的需求 字符串太长了,DB查询效率受影响 数据库自增ID 如果使用 ...

  6. C# 变体(variance)

    上节讲到了泛型,这节延申一下,讲一下变体. 变体(variance)是协变(convariance)和抗变(也说逆变contravariance)的统称.这个概念在.net 4中引入,在.net 2. ...

  7. 分布式锁为什么要选择Zookeeper而不是Redis?

    在分布式的应用中,为了防止单点故障,保障高可用,通常会采用主从结构,当主节点挂掉后,从节点可以代替主节点提供服务. Redis通过复制 + sentinel哨兵来实现主从模式. Zookeeper通过 ...

  8. [bug] Scala eclipse:找不到或无法加载主类

    原因 混合java 和scala 无法编译 解决 右键项目 > properties > scala Compiler >勾选Use Project Setting > Sca ...

  9. [Java] HOW2J(Java初级)

    变量 基本类型:整型(byte.short.int.long).字符型(char).浮点型(float.double).布尔型(boolean) 给基本类型赋值的方式叫字面值 字符的字面值放在单引号中 ...

  10. 【转载】Linux查看PCIe版本及速率【方法】PCIE的X4X8X16 查看 数量 怎么看

    Linux查看PCIe版本及速率   PCIE有四种不同的规格,通过下图来了解下PCIE的其中2种规格   查看主板上的PCI插槽 # dmidecode | grep --color "P ...