我正在使用 JasperReports 生成报告,并假设将其导出为多种格式。但目前我只使用excel报告。
这是我的控制器代码。
InputStream in = reportTemplate.getTemplate(reportInquery.getTemplateFile());
JasperPrint print = JasperFillManager.fillReport(in, null,
new JRResultSetDataSource(reportDao.getReportData(reportInquery)));
resp = HttpConfig.setHeaders(reportInquery, resp);
Exporter exporter = reportOption.getRenderOptions(reportInquery.getFormat(), resp.getOutputStream(), print);
exporter.exportReport();
在我的配置工厂中,excel报告在调用getRenderOptions
方法后配置如下。
public Exporter exporterOptions(OutputStream outputStream, JasperPrint print) {
JRXlsExporter exporter = new JRXlsExporter();
exporter.setExporterInput(new SimpleExporterInput(print));
OutputStreamExporterOutput outputStreamExporterOutput = new SimpleOutputStreamExporterOutput(outputStream);
exporter.setExporterOutput(outputStreamExporterOutput);
SimpleXlsReportConfiguration configuration = new SimpleXlsReportConfiguration();
configuration.setOnePagePerSheet(true);
configuration.setDetectCellType(true);
configuration.setMaxRowsPerSheet(100);
configuration.setRemoveEmptySpaceBetweenColumns(true);
configuration.setRemoveEmptySpaceBetweenRows(true);
exporter.setConfiguration(configuration);
return exporter;
}
要创建JRResultSetDataSource
我在上面OracleCachedRowSet
方法中使用了oracle getReportData
。
public RowSet getReportData(ReportInqueryDTO reportInquery) {
try {
String query = getQueryById(reportInquery.getTemplateId());
Map<String, String> params = new HashMap<>();
params.put("fromDate", reportInquery.getFromDate());
params.put("toDate", reportInquery.getToDate());
params.put("appName", this.applicationName);
RowSet rowSet = namedParameterJdbcTemplate.query(query, params, new ResultSetExtractor<RowSet>() {
@Override
public RowSet extractData(ResultSet resultSet) throws SQLException, DataAccessException {
OracleCachedRowSet rs = new OracleCachedRowSet();
rs.populate(resultSet);
return rs;
}
});
rowSet.beforeFirst();
return rowSet;
} catch (Exception exception) {
log.error("getReportData failed.");
}
return null;
}
我的问题是,excel总是带有第一个记录丢失,其中指向resultSet
的光标。 (例如:这里rowSet.beforeFirst()
表示在第一个索引之前。我错过了这个场景中的第一条记录。)
我几天都遇到了这个问题。我想知道这是否是一个碧玉问题我该怎么解决它。如果无法解决,我想知道如何在第一条记录之前动态添加缓存行集?
我有几个jrxml文件,其中一个在下面添加。每个文件都遵循相同的格式。
<?xml version="1.0" encoding="UTF-8"?>
<!-- Created with Jaspersoft Studio version 6.0.0.final using JasperReports Library version 6.0.0 -->
<!-- 2014-12-30T15:20:10 -->
<jasperReport xmlns="http://jasperreports.sourceforge.net/jasperreports" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://jasperreports.sourceforge.net/jasperreports http://jasperreports.sourceforge.net/xsd/jasperreport.xsd" name="TTT_Call_Transfers_Report" pageWidth="595" pageHeight="842" columnWidth="555" leftMargin="20" rightMargin="20" topMargin="20" bottomMargin="20" uuid="b61676f8-0366-4125-996c-7564d0f77eb4">
<property name="com.jaspersoft.studio.data.defaultdataadapter" value="NewDataAdapter"/>
<style name="Table_TH" mode="Opaque" backcolor="#F0F8FF">
<box>
<pen lineWidth="0.5" lineColor="#000000"/>
<topPen lineWidth="0.5" lineColor="#000000"/>
<leftPen lineWidth="0.5" lineColor="#000000"/>
<bottomPen lineWidth="0.5" lineColor="#000000"/>
<rightPen lineWidth="0.5" lineColor="#000000"/>
</box>
</style>
<style name="Table_CH" mode="Opaque" backcolor="#BFE1FF">
<box>
<pen lineWidth="0.5" lineColor="#000000"/>
<topPen lineWidth="0.5" lineColor="#000000"/>
<leftPen lineWidth="0.5" lineColor="#000000"/>
<bottomPen lineWidth="0.5" lineColor="#000000"/>
<rightPen lineWidth="0.5" lineColor="#000000"/>
</box>
</style>
<style name="Table_TD" mode="Opaque" backcolor="#FFFFFF">
<box>
<pen lineWidth="0.5" lineColor="#000000"/>
<topPen lineWidth="0.5" lineColor="#000000"/>
<leftPen lineWidth="0.5" lineColor="#000000"/>
<bottomPen lineWidth="0.5" lineColor="#000000"/>
<rightPen lineWidth="0.5" lineColor="#000000"/>
</box>
</style>
<subDataset name="Dataset1" uuid="54db10cf-a696-4f7d-b642-96871feb617d">
<property name="com.jaspersoft.studio.data.defaultdataadapter" value="NewDataAdapter"/>
<queryString language="SQL">
<![CDATA[]]>
</queryString>
<field name="DATE_OF_TRANSFER" class="java.sql.Timestamp"/>
<field name="AGENT_ID" class="java.lang.String"/>
<field name="AGENT_NAME" class="java.lang.String"/>
<field name="CTALK_CONTACT_ID" class="java.lang.String"/>
<field name="ASSOCIATED_QUEUE_ID" class="java.lang.String"/>
</subDataset>
<queryString language="SQL">
<![CDATA[]]>
</queryString>
<background>
<band splitType="Stretch"/>
</background>
<title>
<band height="63" splitType="Stretch">
<staticText>
<reportElement x="160" y="16" width="222" height="30" uuid="d89e2553-d3fb-4360-b797-4f69b60938a5"/>
<text><![CDATA[TTT Call Transfers Report]]></text>
</staticText>
</band>
</title>
<detail>
<band height="253" splitType="Stretch">
<componentElement>
<reportElement x="0" y="0" width="555" height="253" uuid="c229d6c3-18b7-4de2-a669-01e11d25c642"/>
<jr:table xmlns:jr="http://jasperreports.sourceforge.net/jasperreports/components" xsi:schemaLocation="http://jasperreports.sourceforge.net/jasperreports/components http://jasperreports.sourceforge.net/xsd/components.xsd">
<datasetRun subDataset="Dataset1" uuid="e0262612-dc81-4d84-abf8-9b52609ba792">
<datasetParameter name="REPORT_DATA_SOURCE">
<datasetParameterExpression><![CDATA[$P{REPORT_DATA_SOURCE}]]></datasetParameterExpression>
</datasetParameter>
</datasetRun>
<jr:column width="40" uuid="a91061c3-7760-43d2-bdc1-091f0e712aaa">
<jr:columnHeader style="Table_CH" height="30">
<staticText>
<reportElement x="0" y="0" width="40" height="30" uuid="6afcdf24-958c-49f4-9ef3-b757ffd1e8de"/>
<text><![CDATA[DATE_OF_TRANSFER]]></text>
</staticText>
</jr:columnHeader>
<jr:columnFooter style="Table_CH" height="30"/>
<jr:detailCell style="Table_TD" height="30">
<textField>
<reportElement x="0" y="0" width="40" height="30" uuid="84a5e4b7-30e0-409c-b2ab-e13c1dd02950"/>
<textFieldExpression><![CDATA[$F{DATE_OF_TRANSFER}]]></textFieldExpression>
</textField>
</jr:detailCell>
</jr:column>
<jr:column width="40" uuid="c7b5a8eb-04b6-4cd6-9328-e1e8ce11a2fc">
<jr:columnHeader style="Table_CH" height="30">
<staticText>
<reportElement x="0" y="0" width="40" height="30" uuid="d7cc290a-6918-4d47-8f55-b43761bee59b"/>
<text><![CDATA[AGENT_ID]]></text>
</staticText>
</jr:columnHeader>
<jr:columnFooter style="Table_CH" height="30"/>
<jr:detailCell style="Table_TD" height="30">
<textField>
<reportElement x="0" y="0" width="40" height="30" uuid="b5f2b8bf-9132-4447-9209-f1fcb3b873c4"/>
<textFieldExpression><![CDATA[$F{AGENT_ID}]]></textFieldExpression>
</textField>
</jr:detailCell>
</jr:column>
<jr:column width="40" uuid="d2427379-2c72-4e33-9b46-ade8727dbed5">
<jr:columnHeader style="Table_CH" height="30">
<staticText>
<reportElement x="0" y="0" width="40" height="30" uuid="bdcb01c5-5073-4a10-b498-e2d1a2cf7c78"/>
<text><![CDATA[AGENT_NAME]]></text>
</staticText>
</jr:columnHeader>
<jr:columnFooter style="Table_CH" height="30"/>
<jr:detailCell style="Table_TD" height="30">
<textField>
<reportElement x="0" y="0" width="40" height="30" uuid="e259ef09-f446-491c-88dd-f7694197e060"/>
<textFieldExpression><![CDATA[$F{AGENT_NAME}]]></textFieldExpression>
</textField>
</jr:detailCell>
</jr:column>
<jr:column width="40" uuid="81fe9b33-57aa-48b2-914b-e3e9b45cf6a1">
<jr:columnHeader style="Table_CH" height="30">
<staticText>
<reportElement x="0" y="0" width="40" height="30" uuid="ef5e379b-65de-4ea0-a5cb-13368efa16f0"/>
<text><![CDATA[CTALK_CONTACT_ID]]></text>
</staticText>
</jr:columnHeader>
<jr:columnFooter style="Table_CH" height="30"/>
<jr:detailCell style="Table_TD" height="30">
<textField>
<reportElement x="0" y="0" width="40" height="30" uuid="5f49f1a9-379a-4d02-9e55-fc6a865aa828"/>
<textFieldExpression><![CDATA[$F{CTALK_CONTACT_ID}]]></textFieldExpression>
</textField>
</jr:detailCell>
</jr:column>
<jr:column width="40" uuid="f6e8b2a6-925b-41b5-9043-1ffa215871f1">
<jr:columnHeader style="Table_CH" height="30">
<staticText>
<reportElement x="0" y="0" width="40" height="30" uuid="b55c7599-a237-43f1-aff5-464717e3c917"/>
<text><![CDATA[ASSOCIATED_QUEUE_ID]]></text>
</staticText>
</jr:columnHeader>
<jr:columnFooter style="Table_CH" height="30"/>
<jr:detailCell style="Table_TD" height="30">
<textField>
<reportElement x="0" y="0" width="40" height="30" uuid="0d2d53d0-1cb2-4d64-a09d-b0eeacca694d"/>
<textFieldExpression><![CDATA[$F{ASSOCIATED_QUEUE_ID}]]></textFieldExpression>
</textField>
</jr:detailCell>
</jr:column>
</jr:table>
</componentElement>
</band>
</detail>
提前致谢。
答案 0 :(得分:6)
请发布您的jrxml文件。
问题可能不在您的代码中。
在Jasper中,数据源中的记录指针由接收它的每个元素递增(因此,例如,如果您在报告中有一个表,并且您将表的数据源设置为该数据源的数据源报告它将跳过第一条记录。如果是这种情况,您必须将报告中的数据源作为参数传递给表格。
更新:
1.从服务器发送数据源作为参数,并用另一个填充报告(可以为空)。
2.在与bean集合相同类型的报告中删除一个新参数,让它命名为DS1&#39;。
3.设置TableDatasource以使用$ P {DS1}参数。
请参阅我的回复 How to show JRBeanCollectionDataSource data with help of Table component?举个例子。
答案 1 :(得分:0)
正如Laura在此answer中提到的,数据源中的记录指针会被接收它的每个元素递增。因为我传递了结果集,所以我认为将结果集再次作为参数传递没有意义。最后,我决定在结果集的开头添加一个空记录。
在上述问题的getReportData
方法中,您可以按照以下方法执行此操作。
RowSet rowSet = namedParameterJdbcTemplate.query(query, params, new ResultSetExtractor<RowSet>() {
@Override
public RowSet extractData(ResultSet resultSet) throws SQLException, DataAccessException {
OracleCachedRowSet rs = new OracleCachedRowSet();
rs.populate(resultSet);
// Have to add a empty row, because jasper is not displaying
// the first row of report.
rs.setReadOnly(false);
rs.beforeFirst();
rs.moveToInsertRow();
int numCol = rs.getMetaData().getColumnCount();
for (int i = 1; i < numCol + 1; i++) {
// Add null inserted row to each column
rs.updateNull(i);
}
rs.insertRow();
rs.beforeFirst();
return rs;
}
});
答案 2 :(得分:0)
我的实际参数是ds。并且List是allDataListPlan(但是缺少1行) 所以我用 dummyList 添加了 dummy 记录 并将其添加到JasperFillManager中。
从服务器发送数据源作为参数,并用不同的数据填充报告(可以为空)。
在与您的bean集合相同类型的报表中声明一个新参数,我们将其命名为“ 虚拟”。
将TableDatasource设置为使用$ P { dummy }参数。
示例:
*ArrayList<ProductionOrder> **dummyList** = new ArrayList<ProductionOrder>();
**dummyList** .add(new ProductionOrder());
JRBeanCollectionDataSource **dummy** = new JRBeanCollectionDataSource(**dummyList**);
JRBeanCollectionDataSource ds = new JRBeanCollectionDataSource(allDataListPlan);
mapForJasperHeader.put("dummy", **dummy**);
mapForJasperHeader.put("ds_for_plan", ds);
String s = VaadinService.getCurrent().getBaseDirectory().getAbsolutePath();
JasperPrint jasperPrint = JasperFillManager.fillReport(s + "\\JasperFiles\\" + "PlanVsActual.jasper",
mapForJasperHeader, **dummy**);*
答案 3 :(得分:0)
正如他们已经说过的,问题可能不在您的代码中。
当您将数据集传递到 ResultSet 时,数据源中的记录指针会增加 1。我通过创建一个具有以下属性的 ResultSet 解决了这个问题:ResultSet.TYPE_SCROLL_INSENSITIVE
、ResultSet.CONCUR_UPDATABLE
。在该方法中,您只能在数据库中执行查询并执行 rs.beforeFirst
。
Connection conn = ...;
Statement stmt = null;
ResultSet rset = null;
try {
stmt = conn.createStatement(ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_UPDATABLE);
rset = stmt.executeQuery(sql);
if(rset != null){
rset.beforeFirst();
return rset;
} else return null;
} catch (SQLException e) {
e.printStackTrace();
}
beforeFirst()
方法将在第一行之前设置记录指针。
所以,你可以这样做:
rs.beforeFirst();
JRResultSetDataSource jrRS = new JRResultSetDataSource(rs);
jp = JasperFillManager.fillReport(jasperFile, parametersJasper,jrRS);