11

这听起来很奇怪,但我想使用 JavaFX 在服务器端生成图表图像。因为 JavaFX 有很好的画布 API 来执行图像转换连接和定位。

特别是我有一个 spring MVC 服务来生成我的图表作为图像。主要问题是如何从方便的 Spring bean 调用 javaFX API。如果我尝试从 java 应用程序运行 javafx 代码(不扩展 javaFX Application 类),我会得到

java.lang.IllegalStateException: Toolkit not initialized

您对如何解决此问题有任何建议/想法吗?

4

3 回答 3

7

因此,经过一些研究,我使用 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);
}

此解决方案可以移植到非 Spring 应用程序。

这是一个单线程解决方案(在我的情况下就足够了),但我认为它可以用于多线程使用(也许使用 RMI 来调用 draw 方法)。

此外,此解决方案在我的 Windows 工作站上“按原样”工作,但在 linux 服务器环境中,应调用一些附加操作:

  1. 您不能在 OpenJDK 上使用 JavaFX(截至 2013 年 8 月) - 必须切换到 Oracle JDK
  2. Java版本必须不低于Java 7u6
  3. 最复杂的——你必须使用虚拟显示来让 JavaFX 在无头环境中运行:

    apt-get 安装 xvfb

    // 然后在应用程序服务器上启动:

    出口显示=“:99”

    启动-停止-守护进程 --start --background --user jetty --exec "/usr/bin/sudo" -- -u jetty /usr/bin/Xvfb :99 -screen 0 1024x768x24


PS 您还可以通过此解决方案在服务器端使用其他 JavaFX 功能(例如,将 html 导出到图像)。

于 2013-08-28T08:33:38.457 回答
3

如果其他人正在寻找这个,这是一种更简单的方法。使用 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链接在上面。

于 2014-11-05T15:01:17.767 回答
0

也许与此解决方案类似的东西会有所帮助?

JavaFX 2.1:工具包未初始化

否则,我会考虑创建一个服务并将图像推送到数据存储并在您的 Spring 应用程序中检索它。

希望至少能提供一点帮助!

于 2013-08-02T05:03:05.620 回答