——从字体原理到工程落地的完整实践指南
一、问题背景:为什么“生僻字”在 pdf 中总是出问题?
在 java 项目中使用 itext / openpdf / flying saucer 生成 pdf,是非常常见的需求,典型场景包括:
- 合同 / 协议生成
- 电子档案
- 报表导出
- 存证类文件
但一旦涉及中文生僻字(如姓名、地名、少数民族用字、古籍用字),就很容易出现以下问题:
- pdf 中显示为
□ - 直接空白
- 只有数字、符号、序号,中文正文消失
- 本机正常,换一台机器就乱码
这些问题并不是 itext 的 bug,而是对 pdf 字体机制 + 中文字符集 理解不足导致的。
二、核心结论先行(非常重要)
java 生成 pdf 要支持中文生僻字,本质上只取决于三件事:
- 字体是否真正包含该字符(glyph 是否存在)
- pdf 是否使用了该字体(css / font-family)
- 字体是否随 pdf 一起嵌入(embedded)
只要其中任意一个不满足,生僻字一定失败。
三、为什么宋体(simsun)无法支持生僻字?
1. simsun 的历史定位
simsun.ttc(宋体)是一个非常早期的中文字库,主要覆盖:
- gb2312 / gbk
- 常用汉字
但它不包含:
- cjk 扩展区 a / b / c / d / e / f
- 大量现代姓名用字(如:𠮷、𡿺)
- 古籍、少数民族汉字
2. 结论
宋体不是“坏字体”,而是“时代局限字体”
它从设计之初就没打算覆盖 unicode 全中文字符集。
因此:
- word 里能显示 ≠ pdf 一定能显示
- 系统里有 ≠ 所有人机器都有
四、真正支持生僻字的字体:noto / 思源系列
1. 推荐字体
在 java pdf 场景中,强烈推荐使用静态版本的 noto 字体:
notosanscjksc-regular.otf
特点:
- 覆盖几乎全部 unicode 中文字符
- 支持 cjk 扩展区
- 免费可商用(sil ofl)
- itext / flying saucer 完全支持
五、常见字体后缀解析(避免踩坑)
后缀 | 含义 | 是否可用于 java pdf |
| truetype font | ✅ |
| opentype font | ✅(推荐) |
| 字体集合 | ⚠️(易出问题) |
| 可变字体 | ❌(不支持) |
| web 字体 | ❌ |
特别警告:variable font(vf)
notosanscjksc-vf.ttf
itext / flying saucer 完全不支持,使用后常见现象是:
- 只有数字、符号
- 中文正文全部消失
- 不报错,非常迷惑
六、java 端正确的字体加载方式
1. 必须使用identity_h
basefont.identity_h
作用:
告诉 pdf 使用 unicode 编码,而不是单字节编码。
2. 强烈建议使用embedded
basefont.embedded
为什么?
not_embedded只是“引用字体名”- 是否能显示,完全依赖打开 pdf 的环境
- 换机器 / linux / docker 必定翻车
你现在能显示,只是环境“碰巧”有该字体
3. 标准代码示例
itextfontresolver fontresolver = renderer.getfontresolver();
fontresolver.addfont(
"notosanscjksc-regular.otf",
basefont.identity_h,
basefont.embedded
);七、最容易被忽略的一点:html / css 才是真正的“使用字体”
1. 一个常见误区
“我在 java 里 addfont 了,为什么中文还不显示?”
原因是:
addfont 只是“注册”,不是“使用”
2. 必须在模板中显式指定font-family
<style>
body {
font-family: "noto sans cjk sc";
font-size: 12pt;
}
</style>关键点:
- 使用 字体内部 family 名称
- 不是文件名
- 建议写在
body,让所有元素继承
如果不写这一步,flying saucer 会退回到内置 latin 字体,中文直接丢失。
八、为什么换成 noto 后,版式可能发生变化?
这是一个正常且不可避免的现象。
原因:
- 不同字体的字宽不同
- 行高(ascent / descent)不同
- 笔画粗细不同
因此可能导致:
- 换行点变化
- 表格溢出
- 分页变化
九、工程级应对方案(必须知道)
1. 固定行高
body {
line-height: 1.4;
}2. 表格列宽避免“卡死”
优先:
width: 25%;
避免:
width: 80px;
3. 字号微调
经验值:
原宋体字号 | noto 建议 |
12pt | 11pt |
10.5pt | 10pt |
十、生僻字验证(上线前必做)
在模板中加入测试文本:
龘 麤 鱻 𠮷 𡿺
全部可见,才算真正支持生僻字。
十一、最终推荐的“标准方案”
生产级、稳定、可迁移的最佳实践
- 字体:
notosanscjksc-regular.otf - 编码:
identity_h - 嵌入:
embedded - css:
body全局font-family - 行高:显式设置
- 表格:避免极限宽度
十二、一句话总结
java 生成 pdf 的生僻字问题,从来不是“渲染问题”,
而是“字体工程问题”。
理解了 字体覆盖范围 + pdf 字体嵌入机制 + css 使用方式,
这个问题可以一次性、永久性解决。
到此这篇关于java生成pdf时该如何正确支持中文生僻字的文章就介绍到这了,更多相关java生成pdf支持中文生僻字内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!
发表评论