当我想要使用springboot fat jar中包含的字体时,我发现了一个奇怪的行为。在本地机器上运行测试加载自 /resources
目录的字体时,它能够完美工作。但是如果我使用maven构建应用并从终端运行,则会收到以下错误信息:
java.io.IOException: Problem reading font data.
at java.awt.Font.createFont0(Font.java:1000)
at java.awt.Font.createFont(Font.java:877)
我尝试寻找解决方案并进行了以下操作:
- 使用不同的字体(有时会出现关于损坏表格等问题的其他错误)
- 构建各种版本的docker openjdk容器以测试行为
- 构建一个ubuntu容器,安装openjdk
- 使用debian镜像,安装fc-cache,并通过以下方式使字体可用
/usr/share/fonts/truetype/europlate
fc-cache -f -v
- 创建一个临时文件,仅确保在jar内部没有访问问题。
- 从终端运行应用程序,其中加载字体也失败了
- 在编辑器中使用该字体,甚至当应用程序从IDE中运行时也可以正常使用(无测试)
方法:
public Font getFont() throws IOException, FontFormatException {
File f = File.createTempFile("dang", "tmp");
assert f != null;
f.delete();
ClassLoader classLoader = getClass().getClassLoader();
Font font = Font.createFont(Font.TRUETYPE_FONT, classLoader.getSystemResourceAsStream("EuroPlate.ttf"));
font.deriveFont(105f);
System.out.println(font.getFontName());
return font;
}
编辑:
集成开发环境(IDE)运行应用程序 -> 正常工作
终端运行应用程序 -> 失败
堆栈跟踪:
java.io.IOException: Problem reading font data.
at java.desktop/java.awt.Font.createFont0(Font.java:1183)
at java.desktop/java.awt.Font.createFont(Font.java:1052)
at components.NumberPlateUtility.getFont(NumberPlateUtility.java:81)
at components.NumberPlateUtility.completeImage(NumberPlateUtility.java:173)
at main.NumberplateClientCommands.one(NumberplateClientCommands.java:63)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.base/java.lang.reflect.Method.invoke(Method.java:566)
at org.springframework.util.ReflectionUtils.invokeMethod(ReflectionUtils.java:223)
at org.springframework.shell.Shell.evaluate(Shell.java:169)
at org.springframework.shell.Shell.run(Shell.java:134)
at org.springframework.shell.jline.InteractiveShellApplicationRunner.run(InteractiveShellApplicationRunner.java:84)
at org.springframework.boot.SpringApplication.callRunner(SpringApplication.java:783)
at org.springframework.boot.SpringApplication.callRunners(SpringApplication.java:773)
at org.springframework.boot.SpringApplication.run(SpringApplication.java:315)
at org.springframework.boot.SpringApplication.run(SpringApplication.java:1242)
at org.springframework.boot.SpringApplication.run(SpringApplication.java:1230)
at main.Main.main(Main.java:20)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.base/java.lang.reflect.Method.invoke(Method.java:566)
at org.springframework.boot.loader.MainMethodRunner.run(MainMethodRunner.java:48)
at org.springframework.boot.loader.Launcher.launch(Launcher.java:87)
at org.springframework.boot.loader.Launcher.launch(Launcher.java:50)
at org.springframework.boot.loader.JarLauncher.main(JarLauncher.java:51)
openjdk
是相同的吗? - Malathi