在涉及注入(inject)和使用反射调用多个类时,我遇到了一个相当有趣的问题。
背景
报告参数定义表包含多行报告。其中一列包含完整的包和类名称。
ReportDef对象包含报表参数定义
ReportRequest 对象包含报告相关信息,例如您需要哪一个月的报告、报告格式(pdf、txt 等)以及数据收集者制作报告所需的其他相关信息。正确的数据库查询。
ReportHandler 这个类是反射调用的。调用适当的 DataCollector,然后根据收集的数据运行报告。 ReportHandler 确实有一个 @Inject 构造函数。
ReportDataCollector 所有 DataCollector 实现的接口(interface)。它包含方法 getReportData。
EligibilityDataCollector 实现 ReportDataCollector。此类的目的是收集与资格报告相关的所有数据,并将内容添加到资格对象中,然后 ReportHandler 使用该对象来运行报告
XXXXXCalcs 处理计算的多个计算类。构造函数出于各种原因进行注入(inject),但主要是为了能够进行适当的数据库调用(通过 DAO 层)
设计
我在背景部分讨论了这一点,但为了让事情变得清楚,流程是这样的
ReportHandler > xxxxDataCollector > xxxxxCalcs
代码
报告处理程序
// get class name based on the Report Def object. The String returned
// from getReportImplementationClass is the full package + class name
Class<?> cls = Class.forName(reportDefs.getReportImplementationClass());
// create new instance of that class
Object obj = cls.newInstance();
// Cast it to ReportDataCollector to ensure that getReportData exists
// note this is in a try / catch and will catch any issues that might arise
// here
ReportDataCollector reportObj = (ReportDataCollector) obj;
// Get the report data needed
birtGroupList = reportObj.getReportData(reportRequest, emProvider);
// Run the report with that data returned
if (birtGroupList != null) {
result = runner.runReport(birtGroupList, reportRequest);
}
资格数据收集器
具体代码无关紧要,但基本上 getReportData 最终协调了所有需要的信息。该信息的一部分是调用计算类。数据收集器知道它需要什么数据,因此它调用特定的计算类。
xxxxx计算
此类有一个注入(inject)构造函数。
问题
到目前为止,报告代码不必调用任何计算类,因此通过反射调用需要注入(inject)的类不是问题。 cls.newInstance() 有效,因此所有内容都通过 ReportHandler 到达 xxxxDataCollector,没有任何使用。
我想要做的事情是通过反射调用 EligibilityDataCollector 构造函数并注入(inject) xxxxCalcs 实例,以便我可以执行我需要的计算。
因此,我需要 ReportHandler 来确定是否存在非空构造函数,然后调用它来提供所需的注入(inject)信息。
或者,如果我可以创建或获取注入(inject)数据,然后调用 xxxxCalcs,也可以在 EligibilityDataCollector 级别上进行操作。
问题之一是我不能只用“new”实例化 xxxCalcs 对象,因为它的构造函数也需要注入(inject),这些对象也有带有注入(inject)的构造函数......所以它很快就会变得非常困惑。这甚至还没有算上我会破坏整个项目中期望注入(inject)的整体设计这一事实。
最佳答案
这个问题的解决方案最终需要一些工作才能弄清楚,但这里是......
我更新了 ReportHandler 类构造函数以接受 Injector 对象。
然后反射部分就更新成了这个
Class<?> cls = Class.forName(reportDefs.getReportImplementationClass());
Object obj = null;
// get the constructors
Constructor<?>[] allConstructors = cls.getDeclaredConstructors();
// must have at least one constructor
if ((allConstructors.length <= 2) && (allConstructors.length > 0)) {
for (Constructor<?> constructor : allConstructors) {
Class<?>[] pType = constructor.getParameterTypes();
if (pType.length >= 1) { // grab the non empty constructor
Object[] objParamsObjects = new Object[pType.length];
for (int i = 0; i < pType.length; i++) {
objParamsObjects[i] = injector.getInstance(pType[i]);
}
obj = constructor.newInstance(objParamsObjects);
break;
}
}
// constructor with no parameters
if (obj == null) {
obj = cls.newInstance();
}
ReportDataCollector reportObj = (ReportDataCollector) obj;
birtGroupList = reportObj.getReportData(reportRequest, emProvider);
if (birtGroupList != null) {
result = runner.runReport(birtGroupList, reportRequest);
}
}
上述实现的唯一问题是您是否期望多个构造函数具有相同数量的参数。在我正在处理的实现中,这种情况不应该发生,但做出假设永远都不好。
关于java - 使用反射调用多个类时的 Guice 注入(inject),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34094316/