情境:
我有两个报告:主要报告(让我们称之为A)和子报告(让我们称之为,B)。
报告A在详细信息带中包含子报告B,因此在报告A数据源中为每个元素显示子报告B.子报告B还将一个变量返回到主报告A。
我想要的是是对子报告B中的返回值求和,并在主报告摘要中对它们进行总计。
为此,我尝试创建一个新的报表变量,将这些返回值相加......就像这样:
但是,我发现在渲染波段细节之前总会评估这样的变量表达式,所以我总是会错过第一个子报表返回值......
可悲的是,评估时间(如link所述)不能在这些变量上改变,所以我被卡住了......
答案 0 :(得分:2)
经过几个小时的努力......并在互联网上寻找解决方案......我带来了一个解决方法(启发性论坛是这些:one和two)。
首先,您需要定义一个java Class Helper,它允许您计算一些算术运算,在我的例子中是一个Sum运算。我定义了这些类:
package reports.utils;
import java.util.Map;
/**
* Utility that allows you to sum Integer values.
*/
public class SumCalculator {
/**
* Stores a map of {@code SumCalculator} instances (A Map instance per thread).
*/
private static final ThreadLocalMap<String, SumCalculator> calculatorsIndex = new ThreadLocalMap<>();
/**
* The sum total.
*/
private int total = 0;
/**
* No arguments class constructor.
*/
private SumCalculator() {
super();
}
/**
* Instance a new {@code SumCalculator} with the given ID.
*
* @param id {@code SumCalculator}'s ID
* @return the new {@code SumCalculator} instance
*/
public static SumCalculator get(String id) {
Map<String, SumCalculator> map = calculatorsIndex.get();
SumCalculator calculator = map.get(id);
if (calculator == null) {
calculator = new SumCalculator();
map.put(id, calculator);
}
return calculator;
}
/**
* Destroy the {@code SumCalculator} associated to the given ID.
*
* @param id {@code SumCalculator}'s ID
* @return {@code null}
*/
public static String destroy(String id) {
Map<String, SumCalculator> map;
map = calculatorsIndex.get();
map.remove(id);
if (map.isEmpty()) {
calculatorsIndex.remove();
}
return null;
}
/**
* Resets the {@code SumCalculator} total.
*
* @return {@code null}
*/
public String reset() {
total = 0;
return null;
}
/**
* Adds the given integer value to the accumulated total.
*
* @param i an integer value (can be null)
* @return {@code null}
*/
public String add(Integer i) {
this.total += (i != null) ? i.intValue() : 0;
return null;
}
/**
* Return the accumulated total.
*
* @return an Integer value (won't be null, never!)
*/
public Integer getTotal() {
return this.total;
}
}
package reports.utils;
import java.util.HashMap;
import java.util.Map;
/**
* Thread Local variable that holds a {@code java.util.Map}.
*/
class ThreadLocalMap<K, V> extends ThreadLocal<Map<K, V>> {
/**
* Class Constructor.
*/
public ThreadLocalMap() {
super();
}
/* (non-Javadoc)
* @see java.lang.ThreadLocal#initialValue()
*/
@Override
protected Map<K, V> initialValue() {
return new HashMap<>();
}
}
其次,在您的jasper报告中,您需要定义四个文本字段:
1)一个文本字段,可以对您的计算器进行初始化;它应该(理想情况下)在报告的标题部分,并且应该有这样的表达式:SumCalculator.get("$V{SUB_REPORT_RETURN_VALUE}").reset()
。此文本字段应具有评估时间:现在。
2)调用增量函数的文本字段(即SumCalculator.get("$V{SUB_REPORT_RETURN_VALUE}").add($V{SUB_REPORT_RETURN_VALUE})
。此文本字段将位于子报表元素之后的详细信息区域;并且它应具有评估时间: BAND(this非常重要!)
3)打印计算器总数的文本字段。此文本字段将位于您的摘要范围内,它将评估为 NOW 。其表达式为:SumCalculator.get("$V{SUB_REPORT_RETURN_VALUE}").getTotal()
4)破坏计算器的文本字段。此文本字段也将位于摘要区域,并且必须显示在文本字段3之后。文本字段应具有如下表达式:SumCalculator.destroy("$V{SUB_REPORT_RETURN_VALUE}")
。此文本字段应具有评估时间:现在。
此外,文本字段:1,2和4应具有属性“Blank at Null”,因此它们永远不会被打印(这就是为什么这些java操作总是返回null)。
就是这样。然后,您的报告可能如下所示:
答案 1 :(得分:0)
如果我理解了这个问题,你无法总结主报告中子报告返回的数量,我遇到了同样的问题,我就这样解决了。
1.-创建一个从 net.sf.jasperreports.engine.JRDefaultScriptlet 扩展的类。并覆盖方法 beforeReportInit()
这是这个类的代码。
C2 A0
2.-在你的ireport的classpath中添加你的项目jar。
3.-在组页脚中添加要打印子报告返回的值的文本字段,其中包含以下表达式。
package com.mem.utils;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import net.sf.jasperreports.engine.JRDefaultScriptlet;
public class SumarizacionSubtotales extends JRDefaultScriptlet {
private final Log log = LogFactory.getLog(getClass());
private Double total;
public Double getTotal() {
return total;
}
public Double add(Double cantidad) {
if(log.isDebugEnabled())log.debug("AGREGANDO LA CANTIDAD : " + cantidad);
this.total += cantidad;
return cantidad;
}
@Override
public void beforeReportInit() throws JRScriptletException {
if(log.isDebugEnabled())log.debug("beforeReportInit");
total = 0.0D;
}
}
在这种情况下, $ V {sum_detalles} 是主报告中的变量,其中包含子报告返回的值。
4.-在“最后一页”页面中添加另一个带有以下表达式的文本字段。
$P{REPORT_SCRIPTLET}.add( $V{sum_detalles} )