将 RHEL 从 7.3 升级到 7.4:sun.font.CompositeStrike.getStrikeForSlot 中的 ArrayIndexOutOfBoundsException

Upgrade RHEL from 7.3 to 7.4: ArrayIndexOutOfBoundsException in sun.font.CompositeStrike.getStrikeForSlot

我们刚刚将服务器从 RHEL v7.3 升级到 v7.4

这个简单的程序在 RHEL v7.3 中有效,在 v7.4 中失败

public class TestJava {
  public static void main(String[] args) {
    Font font = new Font("SansSerif", Font.PLAIN, 12);
    FontRenderContext frc = new FontRenderContext(null, false, false);
    TextLayout layout = new TextLayout("\ude00", font, frc);
    layout.getCaretShapes(0);
    System.out.println(layout);
  }
}

RHEL 7.4 中的例外是:

Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 0
    at sun.font.CompositeStrike.getStrikeForSlot(CompositeStrike.java:75)
    at sun.font.CompositeStrike.getFontMetrics(CompositeStrike.java:93)
    at sun.font.Font2D.getFontMetrics(Font2D.java:415)
    at java.awt.Font.defaultLineMetrics(Font.java:2176)
    at java.awt.Font.getLineMetrics(Font.java:2283)
    at java.awt.font.TextLayout.fastInit(TextLayout.java:598)
    at java.awt.font.TextLayout.<init>(TextLayout.java:393)

在 RHEL v7.3 上的结果是:

sun.font.StandardTextSource@7ba4f24f[start:0, len:1, cstart:0, clen:1, chars:"de00", level:0, flags:0, font:java.awt.Font[family=SansSerif,name=SansSerif,style=plain,size=12], frc:java.awt.font.FontRenderContext@c14b833b, cm:sun.font.CoreMetrics@412ae196]

RHEL v7.4的更新包括openjdk1.8.0.1311.8.0.141的更新,但这似乎与[=17=的版本无关],因为问题与 WebSphere v9.0 (v1.8.0 SR4 FP6) 附带的 IBM JDK 相同。在 RHEL v7.3RHEL v7.4 服务器上使用相同版本的 IBM JDK,该程序在 RH 7.3 中运行,在 RH 7.4 中失败,与 openjdk[=28 相同=]

知道发生了什么吗?

我们终于找到了!
RHEL v7.4(从 v7.3 升级或全新安装)附带软件包 stix-fonts.
安装此包时,默认字体从 Utopia 更改为 STIX 因此,java 现在默认字体映射到 STIX,包括 sans-serif 字体系列
无论出于何种原因,STIX 字体似乎不能很好地与 java (openjdk+ IBM JDK) 一起使用,并在使用 [=19= 时导致异常和错误的计算人工制品定位],JasperReports
就是这种情况 我们以此结束创建文件名 /etc/fonts/local.conf 以强制返回 Utopia 作为默认字体,由 java..

使用
<?xml version='1.0'?>
<!DOCTYPE fontconfig SYSTEM 'fonts.dtd'>
<fontconfig>
  <alias>
    <family>serif</family>
    <prefer><family>Utopia</family></prefer>
  </alias>
  <alias>
    <family>sans-serif</family>
    <prefer><family>Utopia</family></prefer>
  </alias>
  <alias>
    <family>monospace</family>
    <prefer><family>Utopia</family></prefer>
  </alias>
  <alias>
    <family>dialog</family>
    <prefer><family>Utopia</family></prefer>
  </alias>
  <alias>
    <family>dialoginput</family>
    <prefer><family>Utopia</family></prefer>
  </alias>
</fontconfig>

[2018-10-22 编辑]
该错误似乎已在 JDK 1.8.192 中修复:https://bugs.java.com/view_bug.do?bug_id=JDK-8188030

[2019-06-28 编辑]
现在有一个修复程序可以解决 IBM JDK v8.05.37 中包含的问题 http://www-01.ibm.com/support/docview.wss?uid=swg1IJ16655

如果您只安装 "dejavu-serif-fonts" 等任何其他字体,问题将得到解决。或者您使用 "local.conf" 文件来解决问题。 RedHat 已将该问题列为 7.4 中的错误,并且正在寻找真正的解决方案和依赖项:请看这里:https://bugzilla.redhat.com/show_bug.cgi?id=1479563

环境 RHEL 7.4

使用 OpenJDK 或 IBM JDK 升级后,我们无法使用无衬线字体。

在 openjdk 1.8.0.141-2 中使用字体库的新异常。 解析度 该问题已在 java-1.8.0-openjdk-1.8.0.151-1.b12.el7_4.x86_64 中修复。 RHSA-2017:2998 - 安全报告 解决方法

观察到在某些情况下安装 dejavu-serif-fonts 可以解决问题。

yum 安装 dejavu-serif-fonts

如果有人在干净的 Linux 服务器实例上 运行 应用程序遇到问题,运行 命令 fc-list 并验证它是否有多种字体和来自不同的文件夹,就我而言,我在路径 /usr/share/X11/fonts/Type1/ 中有一些额外的字体,我删除了文件夹中的所有内容,问题就解决了。