轻松实现报表集成

0308jc

报表工具作为信息管理、决策分析系统不可或缺的部分,在项目中经常用到,市面上的报表工具各式各样,在集成策略上,每个产品也都有自己特有的打开方式。不过,从门派上看,常见的无外乎中间件,servlet 应用两大门派。一般开源的报表,大都作为中间件产品提供给开发者,集成方法也比较统一,通常都是嵌入到系统中自行调用 API 开发与集成,但开源的帮助文档或全或不全,都靠自己找资料,就要拼学习能力了。在商用收费报表中,润乾报表也采用的是中间件方式,下面我们以润乾报表为例来进一步了解一下:

润乾报表为了方便开发者使用,在满足大多数通用的需求的同时,实现零编程,润乾报表利用 tag 标签进行了封装,如下就是一个发布润乾报表的 jsp 文件:

<%@ taglib uri=”/WEB-INF/raqsoftReport.tld” prefix=”report” %>

<report:html name=”report1″ srcType=”file” reportFileName=”wangge.rpx”…/>

可以看到,在这个 jsp 中用到一个 <prefix: html > 标签来发布 html 报表,与 struts 标签使用类似,其中 reportFileName 就是需要发布的报表文件名,发布报表的同时,实现了导出,分页,打印功能

jc1

如果把 WEB 应用比作一台机器,那报表就是这台机器诸多功能模块里的一块,tag 标签就是一个个小零件,你把这个小零件扣在哪个 jsp 文件中,报表就在哪里展现。
jc2

报表作为中间件集成到其他应用系统里,与这个应用系统就成为了一个整体,但也有分家的做法,比如业界比较常见的 servlet 应用——

jc3

在这里,报表应用是一个独立的应用,在其他应用页面里嵌入一个 JFrame,然后通过 url 请求报表的 servlet,报表名作为参数传递给报表应用,报表应用将计算以后生成的 html 报表返回。

那这个时候,我们的程序猿要问了,浏览报表明白了,那如果报表需要传参呢? 别着急,我们再来模拟一下两种集成方式在给报表传参上有啥区别:

中间件方式

我们还是拿润乾报表的 tag 标签来看:

<%@ taglib uri=”/WEB-INF/raqsoftReport.tld” prefix=”report” %>

<report:html name=”report1″ srcType=”file” reportFileName=”wangge.rpx”

params= arg1=123;arg2=abc;……

/>

Tag 标签中有个params属性,可以通过这个属性给参数赋值,如果参数值灵活变化,当然属性值也可以通过变量传递,这个不多说,程序猿都懂的。

Servlet 的方式

JFrame 中请求报表 servlet 的 URL 就要改成:

http: //ip:port/report/reportservlet ? 参数 = 报表 &arg1=123&arg2=’abc’&…

这时,我知道我们经验丰富的程序猿又要质疑了:首先,引入了 JFrame 来加载报表,多一次数据请求;再次,参数在 url 里都是明文显示的,安全报警啊,特别集成权限时会比较麻烦!除此之外,web 服务与报表服务在不同的进程里,不管是不是在同一台物理服务器,都会多占用硬件资源,当应用出现异常时,排错也麻烦,增加运维难度。所以,润乾报表虽然也支持 servlet 发布报表的方式,但我们也不建议大家这样使用。

前面我们讲了润乾报表作为中间件集成,那么这个中间件到底集成了什么到应用系统中?很简单,就是 jar 包,xml 配置文件及 jsp 的 tag 描述文件 tld。

Jar 包引入到应用程序中,不仅提供了 tag 标签零代码计算展示、打印报表,也为开发者在自己的项目中自行调用报表 API 接口实现报表的计算,分页,导出,打印等过程提供了方便,例如使用代码计算发布一个报表:

// 读报表

String reportFile = C:/reportFiles/wangge.rpx ;// 绝对路径或相对当前程序启动路径

ReportDefine rd = (ReportDefine)ReportUtils.read(reportFile);

// 计算报表

Context cxt = new Context(); // 构建报表引擎计算环境

//…………………….. // 其它辅助代码,例如往报表引擎传递参数,传递数据库连接参数等
ReportDefine rd = (ReportDefine)ReportUtils.read(reportFile);

Engine engine = new Engine(rd, cxt); // 构造报表引擎

IReport iReport = engine.calc(); // 运算报表

// 输出到网页,展现报表

String htmlText = ReportUtils.toHTMLString(iReport,”report1″,request); // 生成 html 语法

request.setAttribute(“htmlText”, htmlText);// 保存到 request 里,方便页面上显示

当然,这里只是讲一个简单例子,API 属于进阶集成,更深入的学习请参考润乾的《程序员参考》文档。这例子虽然简单,但我相信大家已经体会到了她的灵活性、轻量性,而单一的 servlet URL 方式集成如果要二次开发,仅两个系统的数据通讯就是一大难题。 所以基于这种方式集成的报表,要么由开发者想办法解决,要么就是把报表服务做成了一个平台——囊括了数据决策展示、权限管理、报表管理等等看起来牛叉,但跟自己业务系统相结合时又是鸡肋的一堆子业务系统。

便利、灵活、轻量为报表开发节约成本提供了基础,随着业务的提升,高效、高可用性也是考核报表系统的重要部分。因此集群也是集成部署的一个重要环节,那接下来我们继续来看,这两种集成方式在集群上又有啥区别,先上两个结构图——

中间件嵌入到 web 应用里,与 web 应用是一个整体,所以 web 应用做完集群,那么报表也就做完了集群。

jc4

而 Servlet URL 的方式,那集群就相当于分成了两套,各自做自己的集群,也就是报表是一个集群,web 应用是一个集群。

jc5

从以上的部署结构可以看出,中间件的方式在集群上更简单些。所需要的配置及硬件资源都要少不少。

到此报表的集成基本就讲完了,优缺点都做了一个比较,您的应用需要哪一种呢?