这听起来可能很奇怪,但我想使用JavaFX在服务器端生成图表图像。因为JavaFX有很好的画布API来执行图像转换、连接和定位。
特别是,我有一个springMVC服务来将我的图表生成为图像。主要问题是如何从一个方便的Springbean调用javaFX API。如果我尝试只从java应用程序运行javafx代码(不扩展javafx应用程序类),我会得到
java.lang.IllegalStateException: Toolkit not initialized
你对如何解决这个问题有什么建议/想法吗?
因此,经过一些研究,我已经用JavaFX实现了画布绘制,下面是一个简化的示例:
首先,我制作了JavaFX应用程序,它在一个单独的线程中启动(我使用Spring taskExecutor,但可以使用普通的java线程)。
public class ChartGenerator extends Application {
private static Canvas canvas;
private static volatile byte[] result;
public static void initialize(TaskExecutor taskExecutor) {
taskExecutor.execute(new Runnable() {
@Override
public void run() {
launch(ChartGenerator.class);
}
});
}
public static synchronized byte[] generateChart(final Object... params) {
Platform.runLater(new Runnable() {
@Override
public void run() {
ByteArrayOutputStream baos = null;
try {
GraphicsContext gc = canvas.getGraphicsContext2D();
gc.clearRect(0, 0, canvas.getWidth(), canvas.getHeight());
/**
* Do the work with canvas
**/
final SnapshotParameters snapshotParameters = new SnapshotParameters();
snapshotParameters.setFill(Color.TRANSPARENT);
WritableImage image = canvas.snapshot(snapshotParameters, null);
BufferedImage bImage = SwingFXUtils.fromFXImage(image, null);
baos = new ByteArrayOutputStream();
ImageIO.write(bImage, chartType.outputFormat, baos);
result = baos.toByteArray();
} catch (InstantiationException e) {
throw new ChartGenerationException(e);
} catch (IllegalAccessException e) {
throw new ChartGenerationException(e);
} catch (NoSuchMethodException e) {
throw new ChartGenerationException(e);
} catch (InvocationTargetException e) {
throw new ChartGenerationException(e);
} catch (IOException e) {
throw new ChartGenerationException(e);
} finally {
IOUtils.closeQuietly(baos);
}
}
});
while (result == null) {
//wait
}
byte[] ret = result;
result = null;
return ret;
}
@Override
public void start(Stage stage) {
canvas = new Canvas();
}
public static class ChartGenerationException extends RuntimeException {
public ChartGenerationException(String message) {
super(message);
}
public ChartGenerationException(Throwable cause) {
super(cause);
}
}
}
然后,当Spring应用程序启动时,我调用initialize()方法:
@Autowired private TaskExecutor taskExecutor;
@PostConstruct private void initChartGenerator() {
ChartGenerator.initialize(taskExecutor);
}
cource的这个解决方案可以移植到非Spring应用程序。
这是一个单线程解决方案(在我的情况下已经足够了),但我认为它可以用于多线程使用(可能使用RMI来调用draw方法)。
此外,这个解决方案在我的windows工作站上"按原样"工作,但在linux服务器环境中,应该调用一些附加操作:
- 您不能在OpenJDK上使用JavaFX(自2013年8月起)-必须切换到Oracle JDK
- Java版本必须不低于Java 7u6
最复杂的是,您必须使用虚拟显示器才能使JavaFX在无头环境中运行:
apt-get-install xvfb
//然后在应用服务器上启动:
export DISPLAY=":99"
启动-停止守护进程-启动-后台-用户码头-执行"/usr/bin/sudo"--u码头/usr/bin/Xvfb:99-屏幕0 1024x768x24
第页。S.您也可以使用此解决方案在服务器端使用其他JavaFX功能(例如,将html导出为图像)。
如果其他人正在寻找,这是一种简单得多的方法。使用JavaFX 2.2,我能够执行以下操作。
waitForInit = new Semaphore(0);
root = new Group();
root.getChildren().add(jfxnode);
FxPlatformExecutor.runOnFxApplication(() -> {
snapshot = jfxnode.snapshot(new SnapshotParameters(), null);
waitForInit.release();
});
waitForInit.acquireUninterruptibly();
BufferedImage bi = SwingFXUtils.fromFXImage(snapshot, null);
无需将节点添加到组中。从那里你可以对图像进行任何你想要的操作。
FxPlatformExecutor来自我在项目中使用的JME3-JFX库。请参阅:https://github.com/empirephoenix/JME3-JFX/blob/master/src/main/java/com/jme3x/jfx/FxPlatformExecutor.java
您可以轻松地创建runOnFxApplication()
方法或创建FxPlatformExecutor类。
这是代码。
package com.jme3x.jfx;
import javafx.application.Platform;
/**
* TODO This Class should be replaced by some Workmanager implemntation
* in the future
* @author Heist
*/
public class FxPlatformExecutor {
public static void runOnFxApplication(Runnable task) {
if (Platform.isFxApplicationThread()) {
task.run();
} else {
Platform.runLater(task);
}
}
}
我没有写这个代码,github链接在上面。
也许类似于此解决方案的东西会有所帮助?
JavaFX 2.1:工具包未初始化
否则,我会考虑创建一个服务,并将映像推送到数据存储中,然后在spring应用程序中检索它。
希望至少能提供一点帮助!