JasperReports入门教程(二):中文打印
JasperReports入门教程(二):中文打印
背景
在上一篇中我们介绍了JasperReport的基本入门,也展示了一个报表。但是我们的示例都是使用的英文,如果我们把需要打印的数据改为中文会怎么样?下面我们就测试一下
HashMap<String, Object> parameters = new HashMap<String, Object>();
parameters.put("name", "小明");
预览的pdf界面会发现原来打印name的地方不显示任何数据了

为什么会出现这个问题呢?我们查找net.sf.jasperreports的jar包源码发现它并不包含中文的字体库。那么就好办了,我们只需要加上中文字体库就可以了。

给JasperReport Studio工具增加字体
1.我们可以在本地电脑的 C:\Windows\Fonts下找到你想要的字体文件,也可以使用我提供的字体文件微软雅黑msyh.ttf文件下载,提取码: bvmn
2.在JasperReport Studio工具的Window -> Preferences-> font 中add一个微软雅黑的字体

3.修改模板中需要显示中文的元素的字体为微软雅黑

给JasperReport增加扩展的字体资源文件
1.在resources资源文件下增加fonts.xml和msyh.ttf字体文件
- fonts.xml配置微软雅黑的字体
<?xml version="1.0" encoding="UTF-8"?>
<fontFamilies>
<fontFamily name="微软雅黑">
<normal>jaspers/fonts/msyh.ttf</normal>
<bold>jaspers/fonts/msyh.ttf</bold>
<italic>jaspers/fonts/msyh.ttf</italic>
<boldItalic>jaspers/fonts/msyh.ttf</boldItalic>
<pdfEncoding>Identity-H</pdfEncoding>
<pdfEmbedded>true</pdfEmbedded>
<exportFonts>
<export key="net.sf.jasperreports.html">'微软雅黑', Arial, Helvetica, sans-serif</export>
<export key="net.sf.jasperreports.xhtml">'微软雅黑', Arial, Helvetica, sans-serif</export>
</exportFonts>
</fontFamily>
</fontFamilies>
2.在resources资源文件下增加读取扩展字体的配置文件 jasperreports_extension.properties
- jasperreports_extension.properties文件配置如下
net.sf.jasperreports.extension.registry.factory.simple.font.families=net.sf.jasperreports.engine.fonts.SimpleFontExtensionsRegistryFactory
net.sf.jasperreports.extension.simple.font.families.dejavu=jaspers/fonts/fonts.xml
配置好的代码结构如下

3.访问地址 http://localhost:8080/test/jasper?type=pdf 可以看到中文打印出来了

实现原理
下面我们来分析下为什么需要按照上面来配置。因为JasperReport的jar包并未提供微软雅黑的字体的支持,所以我们需要通过提供的字体扩展的接口把我们需要的微软雅黑的字体在程序运行时可以读取到。
其中核心的类就是SimpleFontExtensionsRegistryFactory ,我们查看下这个类的源码
public class SimpleFontExtensionsRegistryFactory implements ExtensionsRegistryFactory
{
public final static String SIMPLE_FONT_FAMILIES_PROPERTY_PREFIX =
DefaultExtensionsRegistry.PROPERTY_REGISTRY_PREFIX + "simple.font.families.";
public final static String PROPERTY_SIMPLE_FONT_FAMILIES_REGISTRY_FACTORY =
DefaultExtensionsRegistry.PROPERTY_REGISTRY_FACTORY_PREFIX + "simple.font.families";
@Override
public ExtensionsRegistry createRegistry(String registryId, JRPropertiesMap properties)
{
List<PropertySuffix> fontFamiliesProperties = JRPropertiesUtil.getProperties(properties, SIMPLE_FONT_FAMILIES_PROPERTY_PREFIX);
List<String> fontFamiliesLocations = new ArrayList<String>();
for (Iterator<PropertySuffix> it = fontFamiliesProperties.iterator(); it.hasNext();)
{
PropertySuffix fontFamiliesProp = it.next();
//String fontFamiliesName = fontFamiliesProp.getSuffix();
String fontFamiliesLocation = fontFamiliesProp.getValue();
//fontFamiliesLocations.addAll(SimpleFontExtensionHelper.getInstance().loadFontFamilies(fontFamiliesLocation));
fontFamiliesLocations.add(fontFamiliesLocation);
}
return new FontExtensionsRegistry(fontFamiliesLocations);
}
}
这个类比较简单只有一个方法,这个createRegistry方法就是读取配置文件中net.sf.jasperreports.extension.simple.font.families开头的配置项的字体文件。
那么追溯这个createRegistry方法在什么地方使用的,可以追溯到类DefaultExtensionsRegistry中的instantiateRegistry方法
查看这个方法可以发现它是加载实现了ExtensionsRegistryFactory接口的某一个类,并且调用instantiateRegistry方法来读取的配置项,这里就是上面提到的SimpleFontExtensionsRegistryFactory 类。
继续向上追溯到类DefaultExtensionsRegistry中的loadRegistries方法,这个方法是加载所有的通过扩展注册进来配置,当然也包括通过扩展注册的字体。
这个方法的代码行较多,我就不贴出来了,那么我们关注下这个方法其中一行代码
List<ClassLoaderResource> extensionResources = loadExtensionPropertyResources();
这行代码是获取所有需要加载的配置文件资源,通过代码我们可以发现它最终是加载所有名称是jasperreports_extension.properties的文件,然后解析文件中的配置项并注册到JasperReport中
protected List<ClassLoaderResource> loadExtensionPropertyResources()
{
return JRLoader.getClassLoaderResources(EXTENSION_RESOURCE_NAME);
}
/**
* The name of property file resources that are used to load JasperReports
* extensions.
*/
public final static String EXTENSION_RESOURCE_NAME = "jasperreports_extension.properties";
到这里我们应该理解上面配置文件jasperreports_extension.properties中的配置项的意义。
##这行配置项是说明需要使用哪个注册工厂类来解析下面的配置项
net.sf.jasperreports.extension.registry.factory.simple.font.families=net.sf.jasperreports.engine.fonts.SimpleFontExtensionsRegistryFactory
##这行配置是配置了需要加载的字体文件的位置,配合上面的```SimpleFontExtensionsRegistryFactory```类来加载字体文件。
net.sf.jasperreports.extension.simple.font.families.dejavu=jaspers/fonts/fonts.xml
结束语
因为方便后续教程的演示,我们把公用的代码包括中文字体的支持都提起到一个公共的模块common,其他模块只引用即可使用,这样可以更好的专注本章节的代码。
JasperReports入门教程(二):中文打印的更多相关文章
- JasperReports入门教程(三):Paramters,Fields和Detail基本组件介绍
JasperReports入门教程(三):Paramter,Field和Detail基本组件介绍 前言 前两篇博客带领大家进行了入门,做出了第一个例子.也解决了中文打印的问题.大家跟着例子也做出了de ...
- JasperReports入门教程(四):多数据源
JasperReports入门教程(四):多数据源 背景 在报表使用中,一个页面需要打印多个表格,每个表格分别使用不同的数据源是很常见的一个需求.假如我们现在有一个需求如下:需要在一个报表同时打印所有 ...
- JasperReports入门教程(一):快速入门
JasperReports入门教程(一):快速入门 背景 现在公司的项目需要实现一个可以配置的报表,以便快速的适应客户的需求变化.后来在网上查资料发现可以使用JasperReports + Jaspe ...
- 无废话ExtJs 入门教程二十一[继承:Extend]
无废话ExtJs 入门教程二十一[继承:Extend] extjs技术交流,欢迎加群(201926085) 在开发中,我们在使用视图组件时,经常要设置宽度,高度,标题等属性.而这些属性可以通过“继承” ...
- 无废话ExtJs 入门教程二十[数据交互:AJAX]
无废话ExtJs 入门教程二十[数据交互:AJAX] extjs技术交流,欢迎加群(521711109) 1.代码如下: 1 <!DOCTYPE html PUBLIC "-//W3C ...
- 无废话ExtJs 入门教程二[Hello World]
无废话ExtJs 入门教程二[Hello World] extjs技术交流,欢迎加群(201926085) 我们在学校里学习任何一门语言都是从"Hello World"开始,这里我 ...
- mongodb入门教程二
title: mongodb入门教程二 date: 2016-04-07 10:33:02 tags: --- 上一篇文章说了mongodb最基本的东西,这边博文就在深入一点,说一下mongo的一些高 ...
- SpringBoot入门教程(二)CentOS部署SpringBoot项目从0到1
在之前的博文<详解intellij idea搭建SpringBoot>介绍了idea搭建SpringBoot的详细过程, 并在<CentOS安装Tomcat>中介绍了Tomca ...
- PySide——Python图形化界面入门教程(二)
PySide——Python图形化界面入门教程(二) ——交互Widget和布局容器 ——Interactive Widgets and Layout Containers 翻译自:http://py ...
随机推荐
- 解决pycharm不能导入bs4模块问题
问题描述: 在导入bs4模块时有报错提示 “ Traceback (most recent call last): File "E:/project/code/py-pengfu/py-pf ...
- Feign客户端实现RPC 调用
1,springcloud 中支持http调用的两种方式,RestTemplate,Feign客户端 2,Feign 客户端是一个声明式(注解方式)http 远程调用工具 3,实现方式如下: 第一步: ...
- SpringCloud微服务架构和SOA架构
1,传统的三层架构 在传统的架构中,SSH,SSM,主要分为web 控制层,业务逻辑层,数据库访问层,单点项目,项目没有拆分,所有的开发任务全部写在一个项目中,耦合度比价高,如果程序中的一个功能出现了 ...
- 将图片base64格式转换为file对象并读取(两种方式读取)
两种方式读取,一种URL.createObjectURL,另一种fileReader var base64 = `  ...
- iOS 状态栏显示/隐藏
iOS 10为止,将在plist文件中将 View controller-based status bar appearance 设置为NO,就可以用UIApplication设置状态栏. 注意,要在 ...
- DFS 蜘蛛纸牌(深度解析)
蜘蛛纸牌 Problem Description 蜘蛛牌是windows xp操作系统自带的一款纸牌游戏,游戏规则是这样的:只能将牌拖到比她大一的牌上面(A最小,K最大),如果拖动的牌上有按顺序排好的 ...
- findbugs过滤R.java文件
在第一次使用findbugs时检查出100多个Bad pratice,仔细一看原来全是R文件里面的类名首字母没有大写导致的. 于是只有自己在findbugs设置界面中添加过滤条件来忽略掉R文件. 在F ...
- python正则表达式去除文本中间的换行符
例如文本: 在这个活动中,请各位同学做一下自我介绍,也看看别的同学的自我介绍,看看谁的介绍更有个性!也许你能在这里找到志趣相投的学习伙伴.也希望大家能够彼此关注,相互鼓励,共同学习. 你的发帖可以包含 ...
- C# 快速开发框架搭建—开发工具介绍
C# 快速开发框架搭建—开发工具介绍 一.VS2013,SQL SERVER R22008 以上两种工具如有不会者自行百度学习下. 二.动软代码生成器 对于经典的三层架构框架来说,使用动软代码生成器会 ...
- 使用 Nginx 实现 301 跳转至 https 的根域名
基于 SEO 和安全性的考量,需要进行 301 跳转,以下使用 Nginx 作通用处理 实现结果 需要将以下地址都统一跳转到 https 的根域名 https://chanvinxiao.com ht ...