我在Java Sprint中有一个API,需要每天执行一项任务,从一个在FTP服务器上生成txt的外部系统导入一些数据。 我遇到的问题是Autowired字段没有自动装配....我的意思是,它们是空的。
每次应用程序启动时,我都在使用@PostConstruct来执行任务,因此我可以使用计时器安排操作。
尝试1
这是代码(首先是PostContruct方法)
@Override
@PostConstruct
@Transactional
public Response importdata(){
Response response = new Response();
try {
System.out.println("*** Setting Import ****");
Calendar calendar = Calendar.getInstance();
calendar.set(Calendar.HOUR_OF_DAY, 21);
calendar.set(Calendar.MINUTE, 5);
calendar.set(Calendar.SECOND, 0);
calendar.set(Calendar.MILLISECOND, 0);
Timer time = new Timer(); // Instantiate Timer Object
time.schedule(new ImportServiceImpl(), calendar.getTime(), TimeUnit.MILLISECONDS.convert(1, TimeUnit.DAYS));
} catch (Exception e) {
e.printStackTrace();
response.setCode(CodeList.EXCEPTION);
response.setSuccess(false);
}
return response;
}
所以在这里,我每天都在21.05安排这个功能。
这里有ImportServiceImpl
@Component
public class ImportServiceImpl extends TimerTask implements ImportService{
@Autowired
InvoiceDao invoiceDao;
@Autowired
ClientDao clientDao;
@Override
@Transactional
public void run() {
System.out.println("*** Running **** " + new Date());
startImport();
}
@Override
@Transactional
public void startImport() {
Path dir = Paths.get(ResourcesLocation.IMPORT_ROUTE);
Boolean success = true;
try (DirectoryStream<Path> stream = Files.newDirectoryStream(dir)) {
for (Path entry : stream) {
if (!Files.isDirectory(entry)) {
BufferedReader br = new BufferedReader(
new InputStreamReader(new FileInputStream(ResourcesLocation.IMPORT_ROUTE + entry.getFileName().toString())));
System.out.println("*** Importing file **** " + ResourcesLocation.IMPORT_ROUTE + entry.getFileName().toString());
try {
String line;
int i = 0;
while ((line = br.readLine()) != null) {
final String[] parts = line.split("\\|");
System.out.println("Line: " + i++ + " Text: " + line);
System.out.println("Factura: " + parts[1]);
Client client = (Client) this.clientDao.get(parts[0]);
String invoiceNumber = this.generateInvoiceNumber(parts[1].substring(1).replace("-", ""));
Invoice inv = (Invoice) this.invoiceDao.getByNumber(invoiceNumber);
if(inv == null){
inv = new Invoice();
inv.setClient(client);
inv.setNumber(invoiceNumber);
inv.setDate(this.convertDate(parts[2]));
inv.setTotal(this.convertFloat(parts[3]));
inv = (Invoice) this.invoiceDao.addOrUpdate(inv);
}
}
}
catch (Exception e) {
e.printStackTrace();
success = false;
}
finally {
try {
br.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
if(success){
try {
Files.delete(entry);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
}
} catch (IOException e1) {
// TODO Auto-generated catch block
e1.printStackTrace();
}
}
这里的问题是this.clientDao应该是自动装配的,但它是null .... 所以我试着做了
if(this.clientDao == null)
this.clientDao = new ClientDaoImpl();
但是,在ClientDaoImpl的get方法中我有
@SuppressWarnings("unchecked")
public Object get(String name) throws Exception
{
Query q = sessionFactory.getCurrentSession()
.createQuery("from " + this.entity + " WHERE name = '" + name + "'");
return q.uniqueResult();
}
并且sessionFactory为null,因为它没有自动装配..我不认为解决方案是继续手动初始化每个类...
尝试2
然后我尝试自动连接类ImportServiceImpl,而不是手动初始化并使用以下命令更改我的代码:
@Autowired
ImportService importService;
和
time.schedule(this.importService, calendar.getTime(), TimeUnit.MILLISECONDS.convert(1, TimeUnit.DAYS));
但是我得到一个错误,因为this.importService不是ImportServiceImpl,它是ImportService,它是接口,接口不能扩展TimerTask。
3尝试
将Autowired类更改为自动装配实施而不是接口。 像这样:
@Autowired
ImportServiceImpl importService;
所以我收到以下错误:
java.lang.IllegalArgumentException: Can not set com.app.services.ImportServiceImpl field com.app.services.InvoiceServiceImpl.importService to com.sun.proxy.$Proxy184
我查看了Why is my Spring @Autowired field null?
中的答案但手动解决方案无效,因为从未设置过上下文。我也尝试使用@Configure注释,这也是在这里建议的,或者我不知道如何使用它或者它不起作用。
简化示例: 我有一个类InvoiceServiceImpl,它有一个带有注释@PostConstruct的方法importdata,因此它在应用程序启动后调用(该部分没问题)importdata方法,为类ImportServiceImpl安排一个时间任务(到目前为止一直很好)。但是当它是正确的时候,该方法被执行但是timertask类中方法内的@Autowired属性为null。
答案 0 :(得分:1)
<强>更新强>
我必须建议对代码的组织方式进行一些小修改。
首先定义ImportService
。
public interface ImportService {
public void startImport();
}
以及相关的实施。
@Service
public class ImportServiceImpl implements ImportService {
@Autowired
private InvoiceDao invoiceDao;
@Autowired
private ClientDao clientDao;
@Override
@Transactional
public void startImport() {
// Process...
}
然后,您实施了TimerTask
。
@Component
public class ImportTimerTask extends TimerTask {
@Autowired
private ImportService importService;
@Override
public void run() {
importService.startImport();
}
}
最后,您在任何类中都有@PostConstruct
方法。
@Autowired
private ImportTimerTask importTimerTask;
@PostConstruct
@Transactional
public void importData() {
Calendar calendar = Calendar.getInstance();
calendar.set(Calendar.HOUR_OF_DAY, 9);
calendar.set(Calendar.MINUTE, 2);
calendar.set(Calendar.SECOND, 0);
calendar.set(Calendar.MILLISECOND, 0);
Timer time = new Timer();
time.schedule(importTimerTask, calendar.getTime(),
TimeUnit.MILLISECONDS.convert(1, TimeUnit.DAYS));
}
通过这样的实现,我的简单测试很好,invoiceDao
和cliendDao
已成功自动装配。
您可以尝试在@EnableAspectJAutoProxy(proxyTargetClass=true)
课程中添加@Configuration
,以便实施尝试3。
您可以找到更多参考资料here。