百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

Compose Desktop 初体验之打包(cpio 打包)

ahcoder 2025-05-10 19:23 1 浏览

从 0 到 1 搞一个 Compose Desktop 版本的玩天气之打包

大家好,前两篇文章大概介绍了下上手 Compose Desktop 和自定义绘制时遇到的一些问题,项目的最终实现效果如下:

视频

代码写好了,该弄的动画也弄了,该请求的网络数据也请求了,该实现的效果也都实现好了,但是!!!咱们得打包出来啊!不打包出来别人如何使用呢?难道说别人想用你开发的桌面应用,结果你给他说你先下载一个 IntelliJ Idae ,然后下载下我的源码,之后把环境配置好,最后运行就可以了!如果下次再想用的时候再运行一次就好了!

这说的是人话嘛,肯定不能这样,所以一定要打包!由于 Compose Desktop 不止可以运行在 Mac 中,还可以运行在 WindowsLinux 中,所以需要打多个包。那使用 Compose Desktop 应该如何打包呢?且听我慢慢道来!

基本用法

插件中的基本配置单元是一个 applicationapplication 是什么呢?在第一篇文章中也提到了,就是在 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>(例如 packageDmgpackageMsi)用于将应用程序打包成相应的格式。这块需要注意的是,目前没有交叉编译支持,因此只能使用特定操作系统构建格式(例如,要构建.dmg 就必须使用 macOS)。默认情况下会跳过与当前操作系统不兼容的任务。
  • packageDistributionForCurrentOS 是一个生命周期任务,聚合了应用程序的所有包任务。
  • packageUberJarForCurrentOS 用于创建单个 jar 文件,其中包含当前操作系统的所有依赖项。
  • run 用于在本地运行应用程序。需要定义一个 mainClass 包含该 main 函数的类。请注意, run 将启动具有完整运行时的非打包 JVM 应用程序。这比创建具有最小运行时间的紧凑二进制映像更快、更容易调试。要运行最终的二进制图像,需要改用 runDistributable
  • createDistributable 用于在不创建安装程序的情况下创建预打包的应用程序映像和最终应用程序映像。
  • runDistributable 用于运行预先打包的应用程序映像。
  • 只有在脚本中使用 blockproperty 时才会创建任务。

光这么说其实有点懵,来一张图大家就明白我说的是什么了!

是不是有点恍然大明白的感觉!直接点击 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 包指定构建版本。

需要注意的是,版本必须遵循以下规则:

  • dmgpkg :格式为 MAJOR.MINOR.PATCH
  • 其中:MAJOR是一个 > 0 的整数;MINOR是一个可选的非负整数;PATCH是一个可选的非负整数;
  • msiexe :格式为 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 块设置特定于平台的选项,使用方法就是上面 maxOSwindowslinux ,不同平台可配置的选项都不太一样!

  • 所有平台
    • 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 格式的图片我们很常见,但是 MacWindows 中的格式是什么鬼。。。没见过啊!

如果知道这两种文件格式的话大家直接跳过后面的部分即可,这里还需要注意的是这里的文件路径指的是项目根目录。

ICNS 文件

  1. 什么是 .icns ?

.icns 是苹果的 macOS 操作系统的 App 图标文件的扩展名,大家在 macOS 的 Desktop 桌面Finder 访达Dock 程序坞 等看到应用程序的外观就是由一个内置在此 App 内部的扩展名为.icns的文件实现的。

可以通过鼠标“右键”点击 App - “显示包内容” - 进入 Contents 目录 - 进入Resources 目录,然后在目录下可以找到名为 Appicon.icns 或其他后缀为 .icns 的一个图标文件。

  1. 如何创建 .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

如果文中写的有误,欢迎在评论区提出,咱们一起探讨。

文章如果能帮助到大家,哪怕是一点,我也非常高兴,先这样。

相关推荐

真快,iOS 16.4 验证已关闭,但仍然可升级

在4月15日早上时段,苹果正式关闭iOS16.4系统验证,意味着你不能通过电脑端进行降级,意思是,你当前系统大于iOS16.4系统版本,你就不能降级了,已经没办法了。亲自测试,使用iPh...

更新后的Linux内核XZ补丁撤销&quot;Jia Tan&quot;作为维护者的身份

今年3月,Linux内核的XZ嵌入式压缩实现项目从公共领域转为BSD零条款许可,并更新了树内代码。此后,在上游XZ项目中又发现了臭名昭著的XZ后门。随着这些重大问题的解决,Las...

CentOS Stream 10发布:Linux 6.12 LTS内核、GNOME 47登场

IT之家12月14日消息,代号为“Coughlan”的CentOSStream10于12月12日正式发布,生命周期大约为五年,将持续维护到2030年。IT之家援引新闻稿,C...

ToDesk Linux更新发布:新增摄像头、网络诊断

ToDesk更新来了!最新版Liunx-4.3.0.0针对Linux用户上新了摄像头、网络诊断等多种实用功能,大幅提升了用户使用体验。并且ToDesk全面支持国产三大操作系统——麒麟、统信、方德,为L...

时隔两年再更新 Linux Skype Alpha发布

【中关村在线软件资讯】7月14日消息:微软在今天面向Linux用户发布了一个全新的Skype版本——LinuxSkypeAlpha,这也是Linux版Skype在2014年来的首次新版更新。这次更...

Linux4.4 RC1释出 75%是驱动更新(linux rtc驱动)

2015-11-1905:35:00作者:鲁畅4.3稳定版发布已半月有余,本周一,Linux创始人LinuxTorvalds在内核邮件中表示,Linux4.4RC1正式释出。早就有Linux爱...

更新后的 DeviceTree 可让微软 Windows Dev Kit 2023 启动 Linux

微软的WindowsDevKit2023也被称为"ProjectVolterra",它是开发人员为改善ARM上的Windows支持而做出的早期努力,开发人员可以在小尺寸ARMP...

腾讯QQ更新Windows 9.9.16、macOS 6.9.58、Linux 3.2.13

IT之家10月20日消息,腾讯QQ于10月18日更新Windows9.9.16、macOS6.9.58、Linux3.2.13,新增支持在图片查看器中识别二维码等功能。新版...

树莓派OS更新:升级至Linux 6.6 LTS,改善对树莓派5支持

IT之家3月14日消息,树莓派基金会(RaspberryPiFoundation)近日更新了树莓派OS(RaspberryPiOS),虽然还是基于DebianGNU/Linu...

腾讯QQ更新Windows 9.9.15、macOS 6.9.55、Linux 3.2.12

IT之家9月29日消息,腾讯QQ更新Windows9.9.15、macOS6.9.55、Linux3.2.12版本,新增支持设置停靠在桌面边缘时不自动隐藏(Windows)等功能...

Linux 5.17 将实现免重启更新主板 BIOS:利用英特尔 PFRUT 技术

IT之家12月30日消息,根据外媒Phoronix消息,英特尔开源项目的工程师已经准备为Linux5.17引入PFRUT技术,实现免重启更新主板BIOS。这项技术属于ACPI...

Fwupd 2.0.9 引入 EFI 证书洞察:Linux 固件更新更安全、更智能

Firmware更新是保持系统安全和优化性能的重要环节。最近,开源工具Fwupd推出了2.0.9版本,为Linux用户带来了更强大的固件更新功能和改进的用户体验!新版本亮点:更清晰、更高...

linux上将大文件切割成小文件之split命令

说明:很多场景需要拷贝或传输文件时,如果我们需要拷贝的文件太大的话,就需要想办法将其分成小个文件进行拷贝,然后载重新合并。今天介绍split命令格式:split[OPTION][INPUT[...

Linux 入门系列——ACL(linux文件acl)

什么是ACLACL是accesscontrollist(访问控制列表)的缩写。主要的目的是提供在传统的owner,group,other的read,write,execute权限之外的更为细的权限...

基于 Linux 快速搭建企业级 Prometheus 监控系统(实战指南)

一、前言随着系统规模的扩大,运维人员需要及时掌握服务器、应用程序、网络等多方面的运行状态,传统监控工具已难以满足现代化需求。Prometheus作为云原生时代最流行的开源监控系统,具备强大的数据采集...