java JasperReports 没有正确读取参数?
声明:本页面是StackOverFlow热门问题的中英对照翻译,遵循CC BY-SA 4.0协议,如果您需要使用它,必须同样遵循CC BY-SA许可,注明原文地址和作者信息,同时你必须将它归于原作者(不是我):StackOverFlow
原文地址: http://stackoverflow.com/questions/3550583/
Warning: these are provided under cc-by-sa 4.0 license. You are free to use/share it, But you must attribute it to the original authors (not me):
StackOverFlow
JasperReports not reading parameters properly?
提问by Rich
I'm working on a web application written in Java using the Tapestry 5.1.0.5 framework. This framework does not have out-of-the-box support for JasperReports, so I wrote a service that modifies ChenilleKit's JasperReport service. I do not depend on the ChenilleKit version, instead I use the JasperReport 3.5.0 dependency. This may not be necessary information, but it never hurts to be specific.
我正在使用 Tapestry 5.1.0.5 框架开发用 Java 编写的 Web 应用程序。这个框架没有对 JasperReports 的开箱即用支持,所以我写了一个修改 ChenilleKit 的 JasperReport 服务的服务。我不依赖于 ChenilleKit 版本,而是使用 JasperReport 3.5.0 依赖项。这可能不是必要的信息,但具体说明也无妨。
Anyway, my service works pretty well. I have it built into the webapp and I am able to compile and output basic reports in PDF, XLS, HTML, and CSV formats. However, I'm having a big issue with getting the SQL in the jasperReport XML file(s) to load the parameter map properly.
无论如何,我的服务运行良好。我已将其内置到 web 应用程序中,并且能够以 PDF、XLS、HTML 和 CSV 格式编译和输出基本报告。但是,我在获取 jasperReport XML 文件中的 SQL 以正确加载参数映射时遇到了一个大问题。
I get the following error when trying to run reports with startdate and enddate parameters.
尝试使用 startdate 和 enddate 参数运行报告时出现以下错误。
SQLException: Missing IN or OUT parameter at index:: 1
SQL knowledge would say this means I have some form of parameter that is not being passed into the SQL. Debug statements indicate to me that I am passing in the parameters alright and that at least some are making their way into the XML report.
SQL 知识会说这意味着我有某种形式的参数没有被传递到 SQL 中。调试语句向我表明我正在传递参数,并且至少有一些正在进入 XML 报告。
For example, I am passing three parameters into a report, Title, StartDate, and EndDate. Title shows up in the rendering of reports, but StartDate and EndDate seem to get lost in translation?
例如,我将三个参数传递到报告中,Title、StartDate 和 EndDate。标题显示在报表的呈现中,但 StartDate 和 EndDate 似乎在翻译中丢失了?
I'm not sure what I'm missing because nearly identical code works in my company's JSP-Tomcat-Servlet based application with JasperReports.
我不确定我错过了什么,因为几乎相同的代码在我公司的基于 JSP-Tomcat-Servlet 的应用程序中使用 JasperReports。
Anyway I'll start showing code and explaining what's going on:
无论如何,我将开始展示代码并解释发生了什么:
public StreamResponse getReport(String reportTitle, ExportFormat formMode, Date startDate, Date endDate) {
Map<String,String> parameterMap = loadParameters(reportTitle);
Connection conn = null;
OutputStream os = new ByteArrayOutputStream();
try{
conn = Report.getConnection();
Resource resc = new ContextResource(cimpl, "src/main/webapp/reports/"+reportTitle+".xml");
log.debug("Calling fillAndExport to fetch the report " + reportTitle);
log.debug("resource="+resc+"\n"+"formMode="+formMode+"\n"+"parameterMap="+parameterMap+"\n"+"conn="+conn+"\n"+
"outputStream="+os);
SimpleDateFormat repDate = new SimpleDateFormat("MM/dd/yyyy HH:mm");
parameterMap.put("StartDate", repDate.format(startDate));
parameterMap.put("EndDate", repDate.format(endDate));
log.debug("StartDate into report: " + parameterMap.get("StartDate"));
log.debug("EndDate into report: " + parameterMap.get("EndDate"));
js.fillAndExport(resc, formMode, parameterMap, conn, os);
SimpleDateFormat sdf = new SimpleDateFormat("MMMddyyyy");
return new JasperStreamResponse((ByteArrayOutputStream) os, formMode, reportTitle+"-"+sdf.format(startDate)+"-"
+sdf.format(endDate));
}catch (Exception e){
log.error("Caught exception while trying to execute the report fillAndExport service method.");
throw new TapestryException("Issue executing report", e);
} finally {
if(conn != null){
try {
conn.close();
} catch (SQLException e) {
System.out.println("Could not close the JDBC connection!!");
}
}
}
}
In a nutshell, I load the report resource, and add the StartDate and EndDate parameters (Title is already in the parameterMap). I then make the call to a JasperService that uses fillAndExport to generate the report. If there are no exceptions it is returned to the browser in a stream.
简而言之,我加载报告资源,并添加 StartDate 和 EndDate 参数(Title 已经在 parameterMap 中)。然后我调用 JasperService,它使用 fillAndExport 来生成报告。如果没有异常,它会以流的形式返回给浏览器。
Here is accompanying debug statements for a given time I get the exception:
这是我收到异常的给定时间随附的调试语句:
[DEBUG] AppModule.ReportService Loaded report class: com.moremagic.reports.TriggerReport
[DEBUG] AppModule.ReportService Calling fillAndExport to fetch the report Trigger
[DEBUG] AppModule.ReportService resource=context:src/main/webapp/reports/Trigger.xml
formMode=HTML
parameterMap={Title=Triggering Merchant Commission}
conn=oracle.jdbc.driver.T4CConnection@7c974e4b
outputStream=
[DEBUG] AppModule.ReportService StartDate into report: 08/22/2010 14:19
[DEBUG] AppModule.ReportService EndDate into report: 08/23/2010 14:19
[DEBUG] AppModule.JasperService Constructed configuration: {}
[DEBUG] AppModule.JasperService Invoking method com.moremagic.services.AppModule.buildJasperService(Logger, Map) (at AppModule.java:188).
[DEBUG] AppModule.JasperService Using template -> src/main/webapp/reports/Trigger.xml
[DEBUG] AppModule.JasperService In fillReport, parameterMap is : {EndDate=08/23/2010 14:31, StartDate=08/22/2010 14:31, Title=Triggering Merchant Commission}
[ERROR] AppModule.JasperService Caught exception in fillReport of ReportsServiceImpl {}
net.sf.jasperreports.engine.JRException: Error executing SQL statement for : WebappReport1
Caused by: java.sql.SQLException: Missing IN or OUT parameter at index:: 1
So as you can see the values are in the parameter map all the way through right until the JasperService calls the JapserReports API. Now I will show the JasperService code and some of the report so you can see the SQL breaking things.
因此,您可以看到这些值一直在参数映射中,直到 JasperService 调用 JapserReports API。现在我将展示 JasperService 代码和一些报告,以便您可以看到 SQL 破坏性的东西。
JasperService:
碧玉服务:
/**
* Fills the report design loaded from the supplied input resource and returns the generated report object.
* <p/>
* if parameter <em>jasperPrint<em> not null, the data filled into <em>jasperPrint</em>
* instead of returns a new jasper print object.
*
* @param jasperPrint
* @param inputResource the input resource (report template file ".jrxml")
* @param parameterMap the parameter map
* @param dataSource the datasource, either a JRDataSource or an SQL JDBC Connection.
*
* @return
*/
public JasperPrint fillReport(JasperPrint jasperPrint, Resource inputResource, Map parameterMap, Object dataSource) throws JRException
{
JasperReport jasperReport = doCompileReportSource(inputResource);
JasperPrint actualJasperPrint;
logger.debug("In fillReport, parameterMap is : " + parameterMap + "\n startDate: " + parameterMap.get("StartDate") + "\n endDate: " + parameterMap.get("EndDate"));
try
{
if (dataSource != null && dataSource instanceof JRDataSource)
actualJasperPrint = JasperFillManager.fillReport(jasperReport, parameterMap, (JRDataSource) dataSource);
else if(dataSource != null && dataSource instanceof Connection)
actualJasperPrint = JasperFillManager.fillReport(jasperReport, parameterMap, (Connection) dataSource);
else
actualJasperPrint = JasperFillManager.fillReport(jasperReport, parameterMap);
if (jasperPrint != null)
{
for (Object page : actualJasperPrint.getPages())
jasperPrint.addPage((JRPrintPage) page);
}
else
jasperPrint = actualJasperPrint;
return jasperPrint;
}
catch (JRException e)
{
logger.error("Caught exception in fillReport of ReportsServiceImpl {}", e);
throw new JRException(e);
}
}
Report:
报告:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE jasperReport PUBLIC "-//JasperReports//DTD Report Design//EN"
"http://jasperreports.sourceforge.net/dtds/jasperreport.dtd">
<jasperReport
name="WebappReport1"
pageWidth="595"
pageHeight="842"
columnWidth="555"
columnSpacing="0"
leftMargin="20"
rightMargin="20"
topMargin="30"
whenNoDataType="AllSectionsNoDetail"
bottomMargin="30">
<reportFont name="Arial_Normal" isDefault="true" fontName="Arial" size="9" pdfFontName="Helvetica" pdfEncoding="Cp1252" isPdfEmbedded="false"/>
<reportFont name="Arial_Bold" isDefault="false" fontName="Arial" size="9" isBold="true" pdfFontName="Helvetica-Bold" pdfEncoding="Cp1252" isPdfEmbedded="false"/>
<reportFont name="Arial_Italic" isDefault="false" fontName="Arial" size="9" isItalic="true" pdfFontName="Helvetica-Oblique" pdfEncoding="Cp1252" isPdfEmbedded="false"/>
<parameter name="Title" class="java.lang.String"/>
<parameter name="StartDate" class="java.lang.String" />
<parameter name="EndDate" class="java.lang.String" />
<!--This is report query string used to fill data-->
<queryString><![CDATA[
SELECT
something
from table b
where
b.ba_timestamp between to_date( $P!{StartDate}, 'MM/DD/YYYY HH24:MI' ) and to_date( $P!{EndDate}, 'MM/DD/YYYY HH24:MI' )+1
]]></queryString>
回答by Guillaume
I think you should either remove the ! in the SQL query or surround your parameters with quotes.
我认为您应该删除 ! 在 SQL 查询中或用引号将参数括起来。
In a jasper query $P{xyz} creates and binds a parameter whereas $P!{xyz} modifies the query string. I think that in your case you're creating an invalid query by appending raw (unquoted) parameters.
在 jasper 查询中,$P{xyz} 创建并绑定一个参数,而 $P!{xyz} 修改查询字符串。我认为在您的情况下,您通过附加原始(未加引号)参数来创建无效查询。
More details in this blog post
这篇博文中有更多细节