用于服务器端图像生成的JavaFX

Nai*_*gun 8 java spring javafx image-processing javafx-2

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

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

java.lang.IllegalStateException: Toolkit not initialized
Run Code Online (Sandbox Code Playgroud)

您有任何建议/想法如何解决这个问题?

Nai*_*gun 6

所以经过一些研究后我用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);
        }
    }

}
Run Code Online (Sandbox Code Playgroud)

然后我在Spring应用程序启动时调用initialize()方法:

@Autowired private TaskExecutor taskExecutor;

@PostConstruct private void initChartGenerator() {
    ChartGenerator.initialize(taskExecutor);
}
Run Code Online (Sandbox Code Playgroud)

这个cource解决方案可以移植到非Spring应用程序.

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

此解决方案在我的Windows工作站上"按原样"工作,但在Linux服务器环境中,应调用一些其他操作:

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

    apt-get install xvfb

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

    export DISPLAY =":99"

    start-stop-daemon --start --background --user jetty --exec"/ usr/bin/sudo" - -u jetty/usr/bin/Xvfb:99 -screen 0 1024x768x24


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