因此,经过一些研究,我使用 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 服务器环境中,应调用一些附加操作:
- 您不能在 OpenJDK 上使用 JavaFX(截至 2013 年 8 月) - 必须切换到 Oracle JDK
- Java版本必须不低于Java 7u6
最复杂的——你必须使用虚拟显示来让 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 导出到图像)。