- 浏览: 547240 次
- 性别:
- 来自: 北京
文章分类
- 全部博客 (740)
- css (4)
- jquery (8)
- javascript (23)
- html (0)
- uml (0)
- 设计模式 (1)
- 开发工具 (14)
- json (4)
- struts 1.x (3)
- spring (3)
- hibernate (6)
- struts 2.x (17)
- JFreechart (0)
- j2se (48)
- jsp (9)
- flex (22)
- 找工作 (1)
- 技术杂谈 (18)
- 网络编程 (5)
- io流 (1)
- ORACLE (15)
- 报表 (3)
- extjs (11)
- jpbm (2)
- swing (5)
- jspereports (3)
- sql (1)
- linux (15)
- ps (1)
- storm (4)
- hbase (8)
- li (0)
- python (1)
- hive (3)
- 机器学习 (1)
- hdfs (1)
- elasticsearch (1)
- hadoop 2.2 (5)
- hadoop (1)
最新评论
-
Tristan_S:
这个有点意思
ASM -
starryskydog:
程序修改detail band部分的样式 如内容字体大小 ...
使用jasperReport实现动态表头 -
samwong:
Good, so usefule
使用YUI Compressor压缩CSS/JS -
gc715409742:
能够告诉我怎么在web项目中使用YUI Compressor? ...
使用YUI Compressor压缩CSS/JS -
JsonTeye:
您好! 我看你的代码,我现在也在做动态报表,实现功能由用户自己 ...
使用jasperreport动态生成pdf,excel,html
最近在我公司drp(运营分销系统)开发中,需要大量报表,由于本人有过jasperReport的开发经验,所以选用了它,jr确实不错,开源,可扩展性很好,缺点就是免费的文档很少,更可气的是,代码中的doc少的可怜,基本上没有参考价值.
由于我们的产品是用于服装行业的,在服装行业有一个尺码组,非常的麻烦,在制作含有尺码组的报表时,表头的各种尺码不能写死,要从数据库查询出来.但是,一般的报表工具都是不支持表头动态化的(我理解,报表嘛,是呈现给特定人物如老板看的特定内容,表头应该是设计好的,不会经常性的更改),jasperReport也是一样,并不直接支持,细究它的实现过程,我们还是可以扩展从而解决这个问题的.
先看jasperReport的流程图.
从上图可看到,jrxml文件要通过JRXmlLoader解析为一个JasperDesign的对象,从源码中可以看出,此对象用java类去描述了报表的整个设计,比如,columnHeader,detail,columnFooter等等.然后由JasperCompileManager编译为一个JasperReport对象,其实,如果你用ireport(jasperReport报表的可视化设计器)制作报表,你完全可以不必理会怎样生成jaserReport对象.ireport对此有很好的支持.
了解了以上过程,我们可以看出,如果要动态的加入设计元素,只能在JasperDesign对象中下手.加入需要的动态元素,我的需求是在columnHeader中加入一个尺码组的表头,代码实现如下.
- package com.webstone.drp.report.common.dynamicHeader;
- import java.io.File;
- import java.lang.reflect.InvocationTargetException;
- import java.util.Iterator;
- import org.apache.commons.beanutils.BeanUtils;
- import net.sf.jasperreports.engine.JRException;
- import net.sf.jasperreports.engine.JasperCompileManager;
- import net.sf.jasperreports.engine.JasperReport;
- import net.sf.jasperreports.engine.design.JRDesignBand;
- import net.sf.jasperreports.engine.design.JRDesignStaticText;
- import net.sf.jasperreports.engine.design.JasperDesign;
- import net.sf.jasperreports.engine.xml.JRXmlLoader;
- /**
- * @author yaer
- */
- @SuppressWarnings("unchecked")
- public class ReportDesignProcess {
- private static final String flagTextKey = "customFlagText";
- public static JasperReport getJasperReport(String xmlFilePath,
- String[][] sizeGroup) throws JRException {
- JasperDesign design = getJasperDesign(xmlFilePath);
- JRDesignBand columnHeader = (JRDesignBand) design.getColumnHeader();
- reSetColumnHeaderHeight(columnHeader, sizeGroup);
- reSetshapeAndPosition(columnHeader, sizeGroup);
- addElementToColumnHeader(columnHeader, sizeGroup);
- return JasperCompileManager.compileReport(design);
- }
- private static JasperDesign getJasperDesign(String filePath)
- throws JRException {
- return JRXmlLoader.load(new File(filePath));
- }
- private static void reSetColumnHeaderHeight(JRDesignBand columnHeader,
- String[][] sizeGroup) {
- columnHeader.setHeight(columnHeader.getHeight() * sizeGroup.length);
- }
- private static JRDesignStaticText getFlagTextInDesign(
- JRDesignBand columnHeader) {
- return (JRDesignStaticText) columnHeader.getElementByKey(flagTextKey);
- }
- private static void reSetshapeAndPosition(JRDesignBand columnHeader,
- String[][] sizeGroup) {
- JRDesignStaticText flagText = getFlagTextInDesign(columnHeader);
- Iterator<jrdesignstatictext></jrdesignstatictext> children = columnHeader.getChildren()
- .iterator();
- JRDesignStaticText element;
- while (children.hasNext()) {
- element = children.next();
- if (element.getX() > flagText.getX()) {
- element.setX(flagText.getX() + flagText.getWidth()
- * sizeGroup[0].length);
- }
- if (!flagTextKey.equals(element.getKey())) {
- element.setHeight(element.getHeight() * sizeGroup.length);
- }
- }
- }
- private static void addElementToColumnHeader(JRDesignBand columnHeader,
- String[][] sizeGroup) {
- JRDesignStaticText flagText = getFlagTextInDesign(columnHeader);
- columnHeader.removeElement(flagText);
- for (int i = 0; i < sizeGroup.length; i++) {
- for (int j = 0; j < sizeGroup[i].length; j++) {
- try {
- JRDesignStaticText newElement = (JRDesignStaticText) BeanUtils
- .cloneBean(flagText);
- newElement.setText(sizeGroup[i][j]);
- newElement.setX(flagText.getX() + flagText.getWidth() * j);
- newElement.setY(flagText.getY() + flagText.getHeight() * i);
- columnHeader.addElement(newElement);
- } catch (IllegalAccessException e) {
- e.printStackTrace();
- } catch (InstantiationException e) {
- e.printStackTrace();
- } catch (InvocationTargetException e) {
- e.printStackTrace();
- } catch (NoSuchMethodException e) {
- e.printStackTrace();
- }
- }
- }
- }
- }
很遗憾,没有写注解,原因是我看了一本书叫<<测试驱动开发>>,里面有一句话"意图导向编程",意思是说,用手段比如容易理解,贴切的类名,方法名,属性达到让读者轻易理解代码.从而少写注解,让代码更简捷.如果大家不大明白以上代码的意思,那就是我写的不够好,还要继续努力.
此类只有一个方法,根据传来的报表文件路径和一个二维数组式的尺码组生成一个jaserReport的对象.有三个关键方法.重新设置columnHeader的height;重新设置静态内容的形状和大小,添加新的元素到columnHeader中,其实,这儿有一个不太容易理的东西:类中有一个flagTextKey的属性,它是标识报表设计中动态内容的一个样板元素,为什么要这个样板元素了,因为用它承载动态内容的样式,要比在用代码实现方便的多.请看BeanUtils.coloneBean()方法,实际上是克隆样板元素对象.
这个类设计的太具体于应用,应该写成一个抽象方法,让子类来具体实现加入动态元素的过程,我相信大家的需求和我不太一样.由于时间关系,我没有仔细考究.毕竟这只是一个参考实现.
最后,在用于ireport画报表时就要注意了,一呈不变的元素该怎么画就怎么画,但样板元素的位置一定要放好.动态内容起始的位置和样式就靠它来定义,大多数时候,它是一个标签.只不过它的"key"属性和上面类的"flagTextKey"要保持一致.
这个话题就到这儿了,我这儿还有一个我包装的工具类,我们公司的同事都认为对开发报表有帮助.
- package com.webstone.drp.report.common;
- import java.util.ArrayList;
- import java.util.Collection;
- import java.util.HashMap;
- import java.util.Map;
- import javax.servlet.http.HttpServletRequest;
- import javax.servlet.http.HttpSession;
- import javax.faces.context.FacesContext;
- import org.apache.commons.lang.ArrayUtils;
- import com.webstone.drp.report.common.dataSource.JRArrayCollectionDataSource;
- import com.webstone.drp.report.common.dynamicHeader.ReportDesignProcess;
- import net.sf.jasperreports.engine.JRException;
- import net.sf.jasperreports.engine.JasperFillManager;
- import net.sf.jasperreports.engine.JasperPrint;
- import net.sf.jasperreports.engine.JasperReport;
- import net.sf.jasperreports.engine.data.JRBeanCollectionDataSource;
- import net.sf.jasperreports.engine.JRDataSource;
- import net.sf.jasperreports.engine.util.JRLoader;
- /**
- * 使用jasperReport做报表时的工具支持类.有两个用途,生成jasperPrint对象,和设置导出时的session
- *
- * @author yaer
- * @date 2006-8-26
- * @modify date 2006-12-8
- */
- public class ReportUtils {
- public static final String XLS = "xls";// 导出为xls文件;
- public static final String PDF = "pdf";// 导出为pdf文件;
- private static final String JASPER = "jasper";// 编译过后的报表文件;
- private static final String JRXML = "jrxml";// 原始的报表文件(xml格式);
- private HttpServletRequest request;
- private HttpSession session;
- private String rootPath;// 报表文件路径
- /**
- * 在jsf环境下时构造些工具类对象
- *
- * @param context
- * jsf的上下文对象
- */
- public ReportUtils(FacesContext context) {
- request = (HttpServletRequest) context.getExternalContext()
- .getRequest();
- session = (HttpSession) context.getExternalContext().getSession(true);
- this.createRootPath(request);// 生成报表文件的绝对路径
- }
- /**
- * 在其它web环境下构造此工具类对象
- *
- * @param req
- * request请求对象
- */
- public ReportUtils(HttpServletRequest req) {
- this.request = req;
- this.session = req.getSession();
- this.createRootPath(request);// 生成报表文件的绝对路径
- }
- /**
- * 获得报表文件的绝对路径
- *
- * @return rootPath
- */
- public String getRootPath() {
- return rootPath;
- }
- /**
- * 获得JasperPrint对象; jasperPrint对象在jasperReport中是填充了报表数据后的一个实体,打印,导出,显示都要使用它.
- * 此方法含有java5.0支持的'可变参数'特性.params其实质是一个对象数组.在调用些方法时要注意它可能的参数顺序.
- * 此方法参数描述:
- * 1、最多只有四个参数。
- * 2、固定参数filePath表示报表文件的路径,为了支持drp系统中动态尺码组做表头的特性, filePath包括两类:
- * 编译过后的文件扩展名为'.jasper'和未编译的原始xml文件'.jrxml';
- * 若报表中有动态尺码组作表头,则filePath为扩展名是'.jrxml'的文件。
- * 若报表中不涉及动态尺码组,则filePath为扩展名是'.jasper'的文件。
- * 3、可变参数params的完整列表是(注意顺序):Object obj/Collection dataSource,String seprator,String[][] sizeGroup.
- * 这三个参数中,有一个例外,Object obj/Collection dataSource必须有一个,此参数表示填充报表的数据,可以是一个Collection式的集合,
- * 也可以是一个model对象(有且只有一个Collection的属性);
- * String seprator表示分隔符,如果数据源是一个Array的集合,则需此参数。String[][]sizeGroup表款尺码组的二维数组。
- *
- * @param filePath
- * @param params
- * @return jasperPrint
- */
- public JasperPrint getJasperPrint(String filePath, Object... params) {
- JasperReport jasperReport = null;
- try {
- if (JASPER.equals(filePath.substring(filePath.indexOf(".") + 1,
- filePath.length()))) {// jasper式文件的处理
- jasperReport = getReportTemplate(filePath);
- }
- if (JRXML.equals(filePath.substring(filePath.indexOf(".") + 1,
- filePath.length()))) {// jrxml式文件的处理
- jasperReport = ReportDesignProcess.getJasperReport(filePath,
- (String[][]) params[params.length - 1]);// 重新设置表头,编译
- params = ArrayUtils.remove(params, params.length - 1);// 删除参数中的sizeGroup
- }
- return fillReport(jasperReport, params);
- } catch (JRException e) {
- e.printStackTrace();
- }
- return null;
- }
- /**
- * 获得JasperPrint对象;自定义填充报表时的parameter和dataSource. 参数说明和动态表头的用法参考上一方法
- * @param filePath
- * @param parameter
- * @param dataSource
- * @param sizeGroup
- * @return
- */
- public JasperPrint getJasperPrint(String filePath, Map parameter,
- JRDataSource dataSource, Object... sizeGroup) {
- JasperReport jasperReport = null;
- try {
- if (sizeGroup.length == 0) {
- jasperReport = getReportTemplate(filePath);
- }
- if (sizeGroup.length == 1) {
- jasperReport = ReportDesignProcess.getJasperReport(filePath,
- (String[][]) sizeGroup[sizeGroup.length - 1]);// 重新设置表头,编译
- }
- return JasperFillManager.fillReport(jasperReport, parameter,
- dataSource);
- } catch (JRException e) {
- e.printStackTrace();
- }
- return null;
- }
- public void setAttrToPage(JasperPrint jasperPrint, String report_fileName,
- String report_type) {
- session.setAttribute("REPORT_JASPERPRINT", jasperPrint);
- session.setAttribute("REPORT_FILENAME", report_fileName);
- session.setAttribute("REPORT_TYPE", report_type);
- }
- private JasperPrint fillReport(JasperReport jasperReport, Object[] params)
- throws JRException {
- Map parameters = null;
- JRDataSource ds = null;
- if (params.length == 0) {
- return null;
- }
- if (params.length == 1 && params[0].getClass() == ArrayList.class) {//其实质是要判断是否是集合
- ds = new JRBeanCollectionDataSource((Collection) params[0]);
- }
- if (params.length == 1 && params[0].getClass() != ArrayList.class) {
- ClassAnalysis ca = new ClassAnalysis(params[0]);
- parameters = ca.getFields();
- ds = new JRBeanCollectionDataSource(ca.getSet());
- }
- if (params.length == 2 && params[0].getClass() == ArrayList.class) {
- ds = new JRArrayCollectionDataSource((Collection) params[0],
- params[1].toString());
- }
- if (params.length == 2 && params[0].getClass() != ArrayList.class) {
- ClassAnalysis ca = new ClassAnalysis(params[0]);
- parameters = ca.getFields();
- ds = new JRArrayCollectionDataSource(ca.getSet(), params[1]
- .toString());
- }
- return JasperFillManager.fillReport(jasperReport,
- parameters == null ? new HashMap() : parameters, ds);
- }
- private void createRootPath(HttpServletRequest request) {
- rootPath = request.getSession().getServletContext().getRealPath("/")
- + "WEB-INF\\classes\\com\\webstone\\drp\\report\\jaser\\";
- }
- private JasperReport getReportTemplate(String jasperPath)
- throws JRException {
- return (JasperReport) JRLoader.loadObject(rootPath + jasperPath);
- }
- }
发表评论
-
JasperReport动态列
2011-10-30 11:59 1696JasperReport动态列 J ... -
使用jasperreport动态生成pdf,excel,html
2011-10-30 11:55 1805使用jasperreport动态生成pdf,excel,ht ... -
Struts Action ActionForm 分类
2010-07-24 19:41 1157Struts Action ActionForm 分类 For ... -
Struts中action跳转种类
2010-07-24 09:39 1144Struts中action跳转种类 关键字: action种类 ...
相关推荐
在本文中,我们将深入探讨如何使用JasperReport来实现动态表头,这是一个在报表生成过程中非常重要的特性,尤其是在处理如服装行业中需要灵活调整的尺码组数据时。JasperReport是一个开源的报表工具,具有良好的可...
在本文中,我们将深入探讨如何使用JasperReport实现动态表头,特别是针对尺码组这样的特殊需求。 1. **JasperReport流程**: JasperReport的工作流程包括XML解析、设计对象构建和编译。首先,JRXML文件被`...
本文将深入探讨如何使用JasperReport实现动态表头,这在处理复杂和多级分类的数据时尤为有用。 JasperReport允许开发者创建交互式、高质量的PDF、HTML、XLS、CSV等多种格式的报表。动态表头是指表头能够根据数据的...
本篇文章将详细讲解如何利用JasperReport实现动态表头和多表分页,并通过subreport来达到灵活的布局效果。 首先,让我们了解JasperReport的基础。JasperReport是一款基于Java的报表设计工具,它允许开发者创建复杂...
同时,合理地使用动态表头和按组分页可以提供更好的用户体验,帮助用户快速理解和解析大量数据。 6. **实战应用**: 在实际项目中,这样的报表设计方法适用于大型企业报表系统,尤其是那些需要处理大量分类数据,...
总的来说,这个项目展示了如何在不使用`subreport`的情况下,利用JasperReport的动态表头和分组功能来创建一个高效、灵活的报表解决方案。对于那些需要处理大量数据和复杂结构的Java应用来说,这样的优化是至关重要...
在这个特定的案例中,我们关注的是如何在JasperReport和iReport中实现固定表头以及隐藏和显示列的功能。 首先,让我们了解一下JasperReport。JasperReport是一个开源的报表库,它允许开发者在各种应用程序中生成...
使用jasperReport实现动态表头.doc详细介绍了如何利用API创建可变的、层次化的表头,这对于处理复杂的数据结构非常有用。 1. JasperReport API:学习如何创建和填充jasperReport模板,理解JRDesign对象和...
这个特定的压缩包文件包含了一个关于如何实现jasperreport动态列以及生成HTML导出的示例,对于使用Eclipse进行Java开发的人员来说,这是一个非常实用的学习资源。 动态列的概念在于,报表的列数不是固定的,而是...
4. **使用`group`**:jasperReport的分组功能也可以帮助实现动态合并。通过设置分组,可以在每个分组的开头和结尾自动插入新行,当数据在同一分组内时,可以利用`printWhenGroupChanges`属性来决定是否合并。 5. **...
在ireport中,可以使用“冻结”或“锁定”功能来实现这一效果。通常,这需要在设计报表时,将表头部分设置为一个独立的表格区域,然后在生成的jrxml文件中配置相应的属性,如`isFixed`,确保在PDF或其他输出格式中...
### JasperReport与iReport的配置与使用 #### 1. JasperReport与iReport简介 ##### 1.1 JasperReport简介 JasperReport是一款强大的报表打印组件,由开放源代码组织sf.net维护。截至本文撰写之时,JasperReport的...
通过报表元素的组合和布局,可以实现灵活的表头结构。 - **交叉表的设计** 交叉表(pivot table)是一种强大的数据分析工具,它能将行和列数据进行重新排列,以显示汇总数据。在iReport中,可以通过向导快速生成...
作为一个全Java编写的库,JasperReport可以很好地集成到Java应用程序、J2EE和Web应用程序中,用于生成动态内容。 JasperReport的主要特点是能够生成面向页面的文档,这些文档通常是为打印准备的。它通过XML文档中的...
在Java开发环境中集成JasperReports可以实现动态生成各种报表的需求。这通常涉及到以下几个步骤: 1. **数据准备**:从数据库或其他数据源中读取数据。 2. **报表设计**:使用JasperReports提供的工具设计报表模板。...
4. 避免冲突:确保合并的单元格不会与其他表格元素(如表头、合计行等)冲突。 在实际应用中,你可能需要结合子报表和行合并功能,例如在一个主报表中展示多个产品的销售数据,并在每个产品下面显示详细的订单行,...
在实际项目中,通常会使用Java程序来动态生成报表,常见的数据源处理方法有: 1. **使用JRBeanCollectionDataSource**: - 在Java程序中查询数据库,将结果封装为Java Bean集合。 - 创建...
- **添加表头**:在columnHeader区使用静态文本工具创建表头。 - **编译报表**:完成设计后,点击"执行报表"按钮,生成.jasper文件。 2. **Java中解析报表**: - **创建Reports类**:在Java代码中,我们需要定义...
开源报表工具JasperReport及其相关组件(iReport和JFreeChart)为开发者提供了一套完整的解决方案,不仅能满足基本的报表需求,还能实现复杂的数据分析与可视化。本文旨在详细介绍这些工具的基本概念、功能特性及...
选择适当的报表类型,如jasperReport或subReport,并为报表命名。 1.3. **创建数据连接** 在报表设计界面,点击“数据”>“连接”来创建新的数据源。输入数据库的相关信息,如URL、用户名、密码,然后选择之前添加...