- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
场景:
我有两个报告:主报告(我们称之为 A)和子报告(我们称之为 B)。
报表 A 在详细信息带中包含子报表 B,因此子报表 B 会针对报表 A 数据源中的每个元素显示。子报表 B 也返回一个变量给主报表 A。
我想要的是对子报表 B 的返回值求和,并在主报表摘要中对它们求和。
为此,我尝试创建一个新的报告变量来对这些返回值求和...像这样:
但是,我发现这样的变量表达式总是在渲染波段细节之前被评估,所以我总是错过第一个子报告返回值......
可悲的是,评估时间(如 link 所说)无法更改这些类型的变量,所以我被困住了......
最佳答案
在为此苦苦挣扎了几个小时之后......并在互联网上搜索解决方案......我想到了一个解决方法(启发性的论坛是这些:one 和 two)。
首先,您需要定义一个 Java 类帮助程序,它允许您计算一些算术运算,在我的例子中是 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(这很重要!!)
3) 打印计算器总数的文本字段。此文本字段将驻留在您的摘要带中,它将评估为现在。其表达式为:SumCalculator.get("$V{SUB_REPORT_RETURN_VALUE}").getTotal()
4) 破坏计算器的文本字段。此文本字段也将位于您的摘要带中,并且必须出现在文本字段 3 之后。文本字段应具有如下表达式:SumCalculator.destroy("$V{SUB_REPORT_RETURN_VALUE}")
。此文本字段应具有评估时间:现在。
此外,文本字段:1、2 和 4 应具有“空时为空”属性,因此永远不会打印它们(这就是为什么那些 java 操作总是返回空)。
就是这样。然后,您的报告可能如下所示:
关于java - 碧 Jade 报告 : How to use subreport return values as input for Main Report Variable Calculation,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28998582/
我是一名优秀的程序员,十分优秀!