自动装配TimerTask无法正常工作

时间:2017-09-21 13:37:59

标签: java spring autowired timertask

我在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。

1 个答案:

答案 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));
}

通过这样的实现,我的简单测试很好,invoiceDaocliendDao已成功自动装配。

您可以尝试在@EnableAspectJAutoProxy(proxyTargetClass=true)课程中添加@Configuration,以便实施尝试3。

您可以找到更多参考资料here