Compose Desktop 初体验之打包(cpio 打包)
ahcoder 2025-05-10 19:23 14 浏览
从 0 到 1 搞一个 Compose Desktop 版本的玩天气之打包
大家好,前两篇文章大概介绍了下上手 Compose Desktop 和自定义绘制时遇到的一些问题,项目的最终实现效果如下:
视频
代码写好了,该弄的动画也弄了,该请求的网络数据也请求了,该实现的效果也都实现好了,但是!!!咱们得打包出来啊!不打包出来别人如何使用呢?难道说别人想用你开发的桌面应用,结果你给他说你先下载一个 IntelliJ Idae ,然后下载下我的源码,之后把环境配置好,最后运行就可以了!如果下次再想用的时候再运行一次就好了!
这说的是人话嘛,肯定不能这样,所以一定要打包!由于 Compose Desktop 不止可以运行在 Mac 中,还可以运行在 Windows 和 Linux 中,所以需要打多个包。那使用 Compose Desktop 应该如何打包呢?且听我慢慢道来!
基本用法
插件中的基本配置单元是一个 application ,application 是什么呢?在第一篇文章中也提到了,就是在 build.gradle.kts 文件中的代码,咱们再来看下:
compose.desktop {
application {
mainClass = "MainKt"
nativeDistributions {
targetFormats(TargetFormat.Dmg, TargetFormat.Msi, TargetFormat.Deb)
packageName = "Demo"
packageVersion = "1.0.0"
}
}
}
一个 application 定义了一组最终二进制文件的共享配置。换句话说,application DSL 允许将一堆文件连同 JDK 分发包打包成一组各种格式(.dmg、.deb、.msi、.exe等)的压缩二进制安装程序。
该插件创建以下任务:
- package<FormatName>(例如 packageDmg 或 packageMsi)用于将应用程序打包成相应的格式。这块需要注意的是,目前没有交叉编译支持,因此只能使用特定操作系统构建格式(例如,要构建.dmg 就必须使用 macOS)。默认情况下会跳过与当前操作系统不兼容的任务。
- packageDistributionForCurrentOS 是一个生命周期任务,聚合了应用程序的所有包任务。
- packageUberJarForCurrentOS 用于创建单个 jar 文件,其中包含当前操作系统的所有依赖项。
- run 用于在本地运行应用程序。需要定义一个 mainClass 包含该 main 函数的类。请注意, run 将启动具有完整运行时的非打包 JVM 应用程序。这比创建具有最小运行时间的紧凑二进制映像更快、更容易调试。要运行最终的二进制图像,需要改用 runDistributable 。
- createDistributable 用于在不创建安装程序的情况下创建预打包的应用程序映像和最终应用程序映像。
- runDistributable 用于运行预先打包的应用程序映像。
- 只有在脚本中使用 block 或 property 时才会创建任务。
光这么说其实有点懵,来一张图大家就明白我说的是什么了!
是不是有点恍然大明白的感觉!直接点击 IntelliJ IDEA 右侧边栏的 Gradle ,就会出现这个侧边栏,然后点击 Task 中的 compose desktop 就会出现上面描述的那些任务。
打包配置
Compose Desktop 打包有很多的配置项,下面来分别看下。
配置包含的 JDK 模块
Gradle 插件使用 jlink 通过仅包含必要的 JDK 模块来最小化可分发的大小。
此时,Gradle 插件不会自动确定必要的 JDK 模块。未能提供必要的模块不会导致编译问题,但会导致在运行时出现 ClassNotFoundException 的错误。
如果在运行打包的应用程序或任务时遇到 ClassNotFoundException ,可以使用DSL 方法runDistributable 来配置包含额外的 JDK 模块,需要使用 modules 来配置。
可以通过手动或运行 suggestModules 任务来确定哪些模块是必需的。suggestModules 使用 jdeps 静态分析工具来确定可能缺少的模块。
如果安装包的大小不重要的话,可以使用 includeAllModulesDSL 属性简单地包括所有运行时模块作为替代。
compose.desktop {
application {
nativeDistributions {
modules("java.sql")
// alternatively: includeAllModules = true
}
}
}
这块在我打包的时候搜了好久!最后在 Issue 中找到了解决方案!
可用格式
以下格式可用于支持的操作系统:
- macOS — .dmg ( TargetFormat.Dmg)、 .pkg( TargetFormat.Pkg)
- Windows — .exe ( TargetFormat.Exe)、 .msi( TargetFormat.Msi)
- Linux — .deb ( TargetFormat.Deb)、 .rpm( TargetFormat.Rpm)
指定包版本
由于可以打多种不同的包,也有可能需要区分不同的版本,所以可以指定包的版本。如何指定的呢?来看代码:
compose.desktop {
application {
nativeDistributions {
packageVersion = "1.0.0"
linux {
packageVersion = "1.0.0"
debPackageVersion = "1.0.0"
rpmPackageVersion = "1.0.0"
}
macOS {
packageVersion = "1.1.0"
dmgPackageVersion = "1.1.0"
pkgPackageVersion = "1.1.0"
packageBuildVersion = "1.1.0"
dmgPackageBuildVersion = "1.1.0"
pkgPackageBuildVersion = "1.1.0"
}
windows {
packageVersion = "1.2.0"
msiPackageVersion = "1.2.0"
exePackageVersion = "1.2.0"
}
}
}
}
必须为本机分发包指定包版本,还可以使用以下 DSL 属性(按优先级降序排列):
- nativeDistributions.<os>.<packageFormat>PackageVersion 指定单个包格式的版本;
- nativeDistributions.<os>.packageVersion 指定单个目标操作系统的版本;
- nativeDistributions.packageVersion 指定所有包的版本;
对于 macOS,还可以使用以下 DSL 属性指定构建版本(按优先级降序排列):
- nativeDistributions.macOS.<packageFormat>PackageBuildVersion 指定单一包格式的构建版本;
- nativeDistributions.macOS.packageBuildVersion 为所有 macOS 包指定构建版本。
需要注意的是,版本必须遵循以下规则:
- dmg 和 pkg :格式为 MAJOR.MINOR.PATCH
- 其中:MAJOR是一个 > 0 的整数;MINOR是一个可选的非负整数;PATCH是一个可选的非负整数;
- msi 和 exe :格式为 MAJOR.MINOR.BUILD
- 其中:MAJOR是一个非负整数,最大值为255;MINOR是一个非负整数,最大值为255;BUILD是一个非负整数,最大值为65535;
- rpm :版本不得包含-(破折号)字符。
- deb :格式为 EPOCH:UPSTREAM_VERSION-DEBIAN_REVISION
- 其中:EPOCH是一个可选的非负整数;UPSTREAM_VERSION 只包含字母数字和字符., +, -, ~,必须以数字开头;DEBIAN_REVISION是可选的,可能只包含字母数字和字符., +, ~。
自定义 JDK 版本
由于该插件使用jpackage,所以最低得使用 JDK 15。
- JAVA_HOME 环境变量指向兼容的 JDK 版本。
- javaHome 通过 DSL 设置:
compose.desktop {
application {
javaHome = System.getenv("JDK_15")
}
}
自定义输出目录
Compose Desktop 默认的打包路径在
/build/compose/binaries/main/app 中,如果想修改下打包路径的话,需要修改下配置:
compose.desktop {
application {
nativeDistributions {
outputBaseDir.set(project.buildDir.resolve("customOutputDir"))
}
}
}
自定义基本数据
DSL 块中提供以下属性 nativeDistributions:
- packageName — 应用程序名称(默认值:Gradle 项目名称);
- version — 应用程序的版本(默认值:Gradle 项目的版本);
- description — 应用程序的描述(默认值:无);
- copyright — 应用程序的版权(默认值:无);
- vendor — 应用程序的供应商(默认值:无);
- licenseFile — 应用程序的许可证(默认值:无)。
compose.desktop {
application {
nativeDistributions {
packageName = "PlayWeather"
version = "1.1.0"
description = "PlayWeather"
copyright = "(c) 2022 My Name. All rights reserved."
vendor = "Example vendor"
licenseFile.set(project.file("LICENSE.txt"))
}
}
}
这块大家可以根据需求来定义这些数据,如不需要不写即可。
特定平台选项
需要使用相应的 DSL 块设置特定于平台的选项,使用方法就是上面 maxOS 、windows 、linux ,不同平台可配置的选项都不太一样!
- 所有平台
- iconFile.set(File("PATH_TO_ICON"))— 应用程序特定于平台的图标的路径。
- packageVersion = "1.0.0" — 特定于平台的包版本。
- installationPath = "PATH_TO_INSTALL_DIR" 默认安装目录的绝对或相对路径;在 Windows 上dirChooser = true,可用于启用在安装过程中自定义路径。
- Linux
- packageName = "custom-package-name" 覆盖默认的应用程序名称;
- debMaintainer = "maintainer@example.com" — deb 包维护者的电子邮件;
- menuGroup = "my-example-menu-group" — 应用程序的菜单组;
- appRelease = "1" — rpm 包的发布值,或 deb 包的修订值;
- appCategory = "CATEGORY" — rpm 包的组值,或 deb 包的部分值;
- rpmLicenseType = "TYPE_OF_LICENSE" — rpm 包的一种许可证;
- debPackageVersion = "DEB_VERSION"``Specifying package version — 特定于 deb 的包版本;
- rpmPackageVersion = "RPM_VERSION"``Specifying package version — 特定于 rpm 的软件包版本;
- MacOS
- bundleID — 唯一的应用标识符;
- 只能包含字母数字字符 ( A-Z, a-z, 0-9)、连字符 ( -) 和句点 ( .) 字符;
- com.mycompany.myapp 建议使用反向 DNS 表示法(例如);
- packageName — 应用名称;
- dockName — 显示在菜单栏、“关于”菜单项、停靠栏等中的应用程序名称。packageName 等于默认情况下的名称;
- signing, notarization, provisioningProfile, 和runtimeProvisioningProfile— 详见相应教程;
- appStore = true — 为 Apple App Store 构建和签名。至少需要 JDK 17;
- appCategory — Apple App Store 的应用类别。默认值是 public.app-category.utilities ;
- entitlementsFile.set(File("PATH_TO_ENTITLEMENTS")) — 包含签名时使用的权利的文件路径;
- runtimeEntitlementsFile.set(File("PATH_TO_RUNTIME_ENTITLEMENTS")) — 包含在签署 JVM 运行时时使用的权利的文件路径;
- dmgPackageVersion = "DMG_VERSION" — 一个特定于 dmg 的包版本(详见参考资料部分);
- pkgPackageVersion = "PKG_VERSION" — 特定于 pkg 的包版本(详情请参阅参考资料部分);
- packageBuildVersion = "DMG_VERSION" — 包构建版本(详见参考资料部分);
- dmgPackageBuildVersion = "DMG_VERSION" — 特定于 dmg 的软件包构建版本(详情请参阅参考资料部分);
- pkgPackageBuildVersion = "PKG_VERSION" — 特定于 pkg 的包构建版本;
- infoPlist — 链接到别的程序。
- Linux
- console = true 为应用程序添加一个控制台启动器;
- dirChooser = true 允许在安装过程中自定义安装路径;
- perUserInstall = true 允许在每个用户的基础上安装应用程序
- menuGroup = "start-menu-group" 将应用程序添加到指定的开始菜单组;
- upgradeUuid = "UUID" — 一个唯一的 ID,当更新的版本比安装的版本更新时,它使用户能够通过安装程序更新应用程序。对于单个应用程序,该值必须保持不变;
- msiPackageVersion = "MSI_VERSION" — 特定于 msi 的软件包版本;
- exePackageVersion = "EXE_VERSION" — 特定于 pkg 的包版本
修改应用图标
这块为什么要单独拿出一大块来说呢?因为这个问题中困扰了我好久。。。所以才。。。。
还记得之前文章中说了 Window 可组合项中可以设定 Icon 么,但当时说的时候专门说了此 Icon 并非应用程序的图标!因为应用程序图标需要以特定于操作系统的格式提供:
- .icns 对于 macOS
- .ico 适用于 Windows
- .png 对于 Linux
看下代码吧:
compose.desktop {
application {
nativeDistributions {
macOS {
iconFile.set(project.file("icon.icns"))
}
windows {
iconFile.set(project.file("icon.ico"))
}
linux {
iconFile.set(project.file("icon.png"))
}
}
}
}
Linux 中的 png 格式的图片我们很常见,但是 Mac 和 Windows 中的格式是什么鬼。。。没见过啊!
如果知道这两种文件格式的话大家直接跳过后面的部分即可,这里还需要注意的是这里的文件路径指的是项目根目录。
ICNS 文件
- 什么是 .icns ?
.icns 是苹果的 macOS 操作系统的 App 图标文件的扩展名,大家在 macOS 的 Desktop 桌面、Finder 访达、Dock 程序坞 等看到应用程序的外观就是由一个内置在此 App 内部的扩展名为.icns的文件实现的。
可以通过鼠标“右键”点击 App - “显示包内容” - 进入 Contents 目录 - 进入Resources 目录,然后在目录下可以找到名为 Appicon.icns 或其他后缀为 .icns 的一个图标文件。
- 如何创建 .icns 扩展名的图标文件?
a. 准备一张图片,重命名为 icon.png,其他大小尺寸可以通过终端命令生成;
b. 通过鼠标右键或者命令,创建一个名为 icons.iconset 的文件夹
mkdir icons.iconset
c. 通过”终端“来快速创建各种不同尺寸要求的图片文件
sips -z 512 512 icon.png -o icons.iconset/icon_512x512.png
d. ”终端“中运行下面的命令,就可以获得名为icon.icns的图标文件了
iconutil -c icns icons.iconset -o icon.icns
注意:icon.png 图片文件和 icons.iconset 文件夹要保存在同一级目录下,”终端“启动后切换到相同目录。
ICO 文件
ico 文件是 Windows 系统的应用图标格式,我也不会制作,但找到了一个制作 ico 的网站:
https://www.butterpig.top/icopro/
需要的话可以进去制作。
混淆
从 Compose Desktop 1.2 版本开始,Compose Gradle 插件支持开箱即用的 ProGuard。ProGuard 是一个众所周知的用于缩小和混淆的开源工具Guardsquare。
Gradle插件为每个对应的默认打包任务提供了发布任务:
默认任务(没有 ProGuard) | 发布任务(带 ProGuard) | 描述 |
createDistributable | createReleaseDistributable | 使用捆绑的 JDK 和资源创建应用程序映像 |
runDistributable | runReleaseDistributable | 使用捆绑的 JDK 和资源运行应用程序映像 |
run | runRelease | jar使用 Gradle JDK运行非打包应用程序 |
package<FORMAT_NAME> | packageRelease<FORMAT_NAME> | 将应用程序映像打包到<FORMAT_NAME>文件中 |
packageForCurrentOS | packageReleaseForCurrentOS | 将应用程序映像打包成与当前操作系统兼容的格式 |
notarize<FORMAT_NAME> | notarizeRelease<FORMAT_NAME> | 上传<FORMAT_NAME>用于公证的应用程序图像(仅限 macOS) |
checkNotarizationStatus | checkReleaseNotarizationStatus | 检查公证是否成功(仅限 macOS) |
默认配置添加了一些 ProGuard 规则:
- 缩小应用程序图像,即删除未使用的类;
- compose.desktop.application.mainClass 用作入口点;
- keep 避免破坏 Compose 运行时的一些规则。
在许多情况下,获得缩小的 Compose 应用程序不需要任何额外的配置。但是,有时 ProGuard 可能无法跟踪字节码中的某些用法(例如,如果通过反射使用类,则可能会发生这种情况)。如果遇到仅在 ProGuard 处理后才会发生的问题,可能需要添加自定义规则。为此,通过 DSL 指定配置文件:
compose.desktop {
application {
buildTypes.release.proguard {
configurationFiles.from(project.file("compose-desktop.pro"))
}
}
}
可以参考 Guardsquare 关于 ProGuard 规则和配置选项的综合手册。
默认情况下禁用混淆。要启用它,请通过 Gradle DSL 设置以下属性:
compose.desktop {
application {
buildTypes.release.proguard {
obfuscate.set(true)
}
}
}
混淆需要根据需要使用了,如果需要控制包体积的话尽量还是打开,可以减小包体积,还会减小代码泄漏的可能性;反之打不打开都行!
实际操作
上面都是理论知识,咱们得实操啊!有基础知识之后实操就很简单了,先来看下 build.gradle.kts 中的配置项吧:
compose.desktop {
application {
mainClass = "MainKt"
nativeDistributions {
targetFormats(TargetFormat.Dmg, TargetFormat.Exe, TargetFormat.Msi, TargetFormat.Deb)
packageName = "PlayWeather"
packageVersion = "1.0.0"
description = "Play Weather App"
copyright = "(c) 2022 My Name. All rights reserved."
vendor = "Lenovo"
licenseFile.set(project.file("LICENSE.txt"))
modules("java.instrument", "java.management", "java.naming", "java.sql", "jdk.unsupported")
linux {
packageVersion = "1.0.0"
debPackageVersion = "1.0.0"
rpmPackageVersion = "1.0.0"
// 设置图标
iconFile.set(project.file("launcher/icon.png"))
}
macOS {
packageVersion = "1.1.0"
dmgPackageVersion = "1.1.0"
pkgPackageVersion = "1.1.0"
dockName = "PlayWeather"
packageBuildVersion = "1.1.0"
dmgPackageBuildVersion = "1.1.0"
pkgPackageBuildVersion = "1.1.0"
// 设置图标
iconFile.set(project.file("launcher/icon.icns"))
}
windows {
packageVersion = "1.2.0"
msiPackageVersion = "1.2.0"
exePackageVersion = "1.2.0"
// 设置图标
iconFile.set(project.file("launcher/icon.ico"))
}
}
buildTypes.release.proguard {
obfuscate.set(false)
configurationFiles.from(project.file("proguard-rules.pro"))
}
}
}
这里的配置项就不多说了,上面都有过介绍,下面来打包吧!
点击上图中蓝色箭头标注的进行打包,上面说过了,不能跨系统打包,Mac 只能打 Mac 中使用的包。。双击执行 packageDmg 任务:
没问题的话大概会出现上图的样子,由于没有配置自定义包路径,所以还在默认文件中,按照上面所描述的路径进行查看:
复制文件路径,在访达中打开:
双击进行安装即可:
然后在资源库中找到应用双击打开应该会遇到下面的错误:
这时点击取消,然后打开设置 -> 隐私与安全性,往下滑:
点击箭头标注的“仍要打开”按钮,会让你输入电脑的密码,输入完成后会弹出下面的对话框:
点击打开,这时应用就能正常使用了。
苹果端就不打 release 包了,还需要苹果的开发者账号那一大堆。。。目前先能正常在 Mac 中运行吧!
总结
本文大概总结了下使用 Compose Desktop 如何进行打包,此项目所有代码都放到了 Github 中。
Github 地址:
https://github.com/zhujiang521/PlayWeather/tree/desktop
如果文中写的有误,欢迎在评论区提出,咱们一起探讨。
文章如果能帮助到大家,哪怕是一点,我也非常高兴,先这样。
相关推荐
- Nginx的http块sendfile,keepalive_timeout的配置指令说明,
-
sendfile是用来设置Ngnix服务的一种静态资源的写法sendfile:用来设置Nginx服务器是否使用sendfile()传输文件,该属性可以大大提高Nginx处理静态资源的性能这里send...
- Docker安装和配置Nginx(docker-compose部署nginx)
-
下载镜像dockerpullnginx检查Nginx镜像dockerimages创建挂载目录mkdir-p/home/docker-nginx/confmkdir-p/home/doc...
- 记:Nginx 如何配置SSL证书(nginx添加ssl证书)
-
引言最近,有几个其他项目团队的同事问了我几个类似问题:“钦哥,我们微信小程序微信支付,怎么回调不没反应?”“钦哥,我们的项目怎么无法通过https访问的?”然后就有想把配置ssl证书的过程,记录一...
- 多项目nginx部署配置(nginx多站点配置示例)
-
在前后端分离的项目中,前端部署大部分都采用nginx部署。在我参与的一个项目中,后台管理前端和业务应用前端是作为两个独立项目开发的。以往部署时通过不同端口来访问这两个项目,这就导致前端存储的token...
- NGINX最佳缓存配置策略(nginx最佳缓存配置策略有哪些)
-
Nginx可以配置的参数有很多,今天我们来看一下Nginx的缓存如何配置。no-cache和no-store“no-cache”表示必须先与服务器确认返回的响应是否发生了变化,然后才能使用该响应来满足...
- 通过 docker 学习 nginx,附全部配置及 API 测试,使用 apifox 直接打开
-
本篇文章以前端的视角,介绍下nginx的常见配置,并通过docker的方式学习nginx,这保证所有实例配置都能正常运行。我将本文所有关于docker/nginx的配置放置在simple...
- nginx配置负载均衡 以及宕机了后,怎么配置高可用
-
Nginx负载均衡配置是一个分发请求的过程。可以通过将同一个服务的多个实例配置到不同的服务器上,通过Nginx代理请求,将请求分发到这些实例上实现负载均衡。下面详细介绍Nginx负载均衡的配置过程和宕...
- nginx反向代理加https证书和自动跳转配置
-
nginx反向代理加https证书和自动跳转配置如果对运维课程感兴趣,可以在b站上搜索我的账号:运维实战课程,可以关注我,学习更多免费的运维实战技术视频1.机器规划:nginx代理机器:192.16...
- 【Nginx基础】Nginx常见负载均衡策略配置示例
-
下面是一些常用的Nginx负载均衡策略及其配置示例:轮询(RoundRobin)这是默认的负载均衡策略,它将请求依次分配给不同的后端服务器。upstreambackend{server...
- nginx中配置固定的JSON响应(nginx指定conf)
-
前提在进行性能测试时,使用mock接口是一种非常有效的方法,可以帮助您模拟真实服务的行为,从而避免对外部系统的依赖。使用Nginx作为mock接口的返回源不仅可以快速配置,还能确保mock接口本身不...
- Nginx基本安全配置(防盗连、隐藏版本号、防嵌套等)
-
在生产环境中nginx有很多安全方案,我在为生产环境部署中得到很多经验,收集到的一些常用配置,简单记录一下。安装Nginx正常安全Nginx大家应该都会,但是各位要知道在安全要求很高的生产环境中,是无...
- nginx如何配置跨域(nginx配置解决跨域)
-
要在Nginx中配置跨域,可以使用add_header指令来添加Access-Control-Allow-*头信息,如下所示:location/api{if($reques...
- Nginx合集-限流配置方案参考(nginx lua限流)
-
Nginx为我们提供了请求限制模块(ngx_http_limit_req_module)、基于令牌桶算法的流量限制模块(ngx_stream_limit_conn_module),可以方便的控制令牌速...
- 韩国站群服务器Nginx反向代理多IP配置指南
-
在站群服务器的运营中,Nginx反向代理技术如同一座智能调度中心,既能隐藏真实服务器架构,又能实现流量精准分发。尤其对于部署在韩国数据中心的站群而言,合理配置多IP反向代理,可显著提升SEO安全性与服...
- Nginx配置前后端服务(nginx配置前端页面部署)
-
nginx安装完成后,可以通过命令查看配置文件nginx-t配置文件nginx.conf,是总的配置,有的人会把配置全部配置到这个文件中,如果服务很多,这个文件变得非常庞大,我见过一个配置很大的,在...
- 一周热门
- 最近发表
-
- Nginx的http块sendfile,keepalive_timeout的配置指令说明,
- Docker安装和配置Nginx(docker-compose部署nginx)
- 记:Nginx 如何配置SSL证书(nginx添加ssl证书)
- 多项目nginx部署配置(nginx多站点配置示例)
- NGINX最佳缓存配置策略(nginx最佳缓存配置策略有哪些)
- 通过 docker 学习 nginx,附全部配置及 API 测试,使用 apifox 直接打开
- nginx配置负载均衡 以及宕机了后,怎么配置高可用
- nginx反向代理加https证书和自动跳转配置
- 【Nginx基础】Nginx常见负载均衡策略配置示例
- nginx中配置固定的JSON响应(nginx指定conf)
- 标签列表
-
- linux 远程 (37)
- u盘 linux (32)
- linux 登录 (34)
- linux 路径 (33)
- linux 文件命令 (35)
- linux 是什么 (35)
- linux 界面 (34)
- 查看文件 linux (35)
- linux 语言 (33)
- linux代码 (32)
- linux 查看命令 (33)
- 关闭linux (34)
- root linux (33)
- 删除文件 linux (35)
- linux 主机 (34)
- linux与 (33)
- linux 函数 (35)
- linux .ssh (35)
- cpu linux (35)
- 查看linux 系统 (32)
- linux 防火墙 (33)
- linux 镜像 (34)
- linux ip地址 (34)
- linux 用户查看 (33)
- nginx配置 解析 (37)