我正在尝试以.jar和.war格式部署Clojure应用程序。到目前为止,我已经成功部署了一个.jar;我正在研究 .war。该应用程序只是一个测试应用程序,旨在演示Clojure和JavaFX的一些功能。我的意思是通过 AWS Elastic Beanstalk 将其部署到我刚刚开始设置的网站。目前,该应用程序只是显示一堆模糊的圆圈,在600x600舞台上移动40秒。但是,即使应用程序在这 40 秒内成功运行,它也总是在这 40 秒的动画之后突然终止。我的 project.clj 文件如下所示:
(defproject clojurefx/arg "0.0.10-SNAPSHOT"
:description "Helper functions and probably a wrapper to simplify usage of JavaFX in Clojure.
This is meant to be used with Java 8.
..."
:url "https://www.github.com/zilti/clojurefx" ; my project is essentially forked from this project
:lein-release {:deploy-via :clojars}
:license {:name "Eclipse Public License"
:url "http://www.eclipse.org/legal/epl-v10.html"}
:dependencies [[org.clojure/clojure "1.5.1"]]
:plugins [[lein-marginalia "0.7.1"]
...]
:profiles {:dev {:dependencies [[midje "1.6-beta1"]]}}
:main clojurefx.arg.splashpage
:aot [clojurefx.arg.splashpage])
这是-main
函数所在的位置:
(ns clojurefx.arg.splashpage
(:require
[clojure.reflect :as clj-r]
[clojure.stacktrace :as stacktrace]
[clojurefx.arg.core :as core]
[clojurefx.arg.core :refer [deffx]]
...)
(:import
(javafx.scene.text Font FontPosture FontWeight Text)
...)
(:gen-class))
(defn -main [& args]
(println "Up and running! The arguments supplied were: " args))
;=====JavaFX visual objects initialized and displayed with the code below=====
;-----Root-----
(deffx rt group)
;-----Scene/Window-----
(deffx scn scene
:width 600
:height 600
:root rt
:fill (. Color BLACK))
;-----Stage/Window-surround-----
(deffx stg stage
:title "Colorful Circles"
...
(dofx (.show stg)) ; shows the JavaFX window with the circle animation
(dofx (.play tmln)) ; play 40s of animation
正如我所说,动画在 40 秒内播放正常,但此时我得到以下异常:
-> Exception in thread "main" java.lang.ExceptionInInitializerError
-> at clojure.main.main(main.java:37)
... ; a long list of traced errors
-> Caused by: java.lang.IllegalStateException: Toolkit not initialized
... ; a long list of traced errors
-> Exception in thread "Thread-4" clojure.lang.ExceptionInfo: Subprocess failed {:exit-code 1}
... ; a long list of traced errors
-> at java.lang.Thread.run(Thread.java:724)
我认为这可能与JavaFX线程的工作方式有关。当动画停止运行时,也许会有某种超时。相比之下,当我注释掉main clojurefx.arg.splashpage
和:aot [clojurefx.arg.splashpage])
时,没有超时project.clj
,只需手动将splashpage.clj
文件中的代码复制、粘贴和评估到 REPL 中。此外,当我将用于 JavaFX 对象初始化和显示的代码放在函数中时,比如说,(init-stage)
,这样在显式调用函数之前不会在运行时评估代码,"超时"在启动 JVM 并调用(-main)
函数后立即开始。
思潮?我意识到了解 Clojure 的人群和了解 JavaFX 8 的人群的交集非常小......
附加信息:
我还有另一个相关的问题。现在,当我尝试部署.jar文件时,该文件已成功创建,如下所示:
Alexanders-MacBook-Pro:cljfx-arg alexandergunnarson$ lein uberjar
Compiling clojurefx.arg.splashpage
Created /Users/alexandergunnarson/.lein/cljfx-arg/target/arg-0.0.10-SNAPSHOT.jar
Created /Users/alexandergunnarson/.lein/cljfx-arg/target/arg-0.0.10-SNAPSHOT-standalone.jar
但是当我在Mac上运行.jar(无论是独立还是非独立)时,OS X显示"无法启动文件"。此外,实际创建.jar的点是动画停止或我退出显示动画的main
应用程序(JVM)的点。
答案是,对于要导入的某些 JavaFX 类,需要调用以下内容:
(defonce force-toolkit-init (javafx.embed.swing.JFXPanel.))
该defonce
用于确保工具包不会以某种方式重新初始化。
即使在导入任何 JavaFX 类之前,也可能需要声明这一点。