自动布线任务发送到弹簧任务执行器

2022-09-03 14:28:01

你怎么能有一个实现Runnable并提交到spring TaskExecutor自动连接的类?

例如,我有一个任务:

public class MyTask implements Runnable {

    @Autowired private MyRepository myRepository;

    @Override
    public void run() {
        myRepository.doSomething();
    }
}

以及向 spring TaskExecutor 发送任务的服务:

@Service
public class MyService {

    @Autowired private TaskExecutor taskExecutor;

    public void someMethod() {

        MyTask myTask = new MyTask();
        taskExecutor.execute(myTask);

    }

}

我知道这些字段没有被自动连接,因为MyTask正在使用新的MyTask()进行实例化。但是,我该如何解决这个问题?我应该访问Spring的AppplicationContext并通过它创建Bean吗?在 Web 应用程序环境中,您将如何执行此操作?

谢谢!


答案 1

尝试

public class MyTask implements Runnable {
    private MyRepository myRepository;

    public MyTask(MyRepository myRepository) {
         this.myRepository = myRepository;
    }

    @Override
    public void run() {
        myRepository.doSomething();
    }
}

@Service
public class MyService {
    @Autowired private TaskExecutor taskExecutor;
    @Autowired private MyRepository myRepository;


    public void someMethod() {
        MyTask myTask = new MyTask(myRepository);
        taskExecutor.execute(myTask);
    }
}

或者你可以声明MyTask的范围=“原型”,并将MyService更改为

@Service
public class MyService {
    @Autowired private ApplicationContext ctx;

    public void someMethod() {
        MyTask myTask = ctx.getBean(MyTask.class);
        taskExecutor.execute(myTask);
    }
}

答案 2

使用Spring至少有两种好方法可以做到这一点。首先,@Configurable注释。使用这意味着对AspectJ的依赖,但它将允许您注入不受Spring管理的bean(即您正在使用新的运算符)。这将涉及使用@Configurable注释MyTask,并在链接中提到的Spring配置中添加几行。

@Configurable
public class MyTask implements Runnable { ... }

@Service
public class MyService {
   @Autowired private TaskExecutor taskExecutor;

   public void someMethod() {

     // AspectJ would jump in here and inject MyTask transparently
     MyTask myTask = new MyTask();
     taskExecutor.execute(myTask);

}

}

第二种方法将涉及使用Spring的ServiceLocatorFactoryBean功能来创建原型bean。这在JavaDoc中得到了最好的解释,但是在这种情况下,您会将TaskFactory注入到@Service注释类中,就像任何其他bean一样,然后执行如下操作:

@Service
public class MyService {
  @Autowired private TaskExecutor taskExecutor;
  @Autowired private MyRepository myRepository;
  @Autowired private TaskFactory taskFactory;


public void someMethod() {
    MyTask myTask = taskFactory.getTask("myTask")
    taskExecutor.execute(myTask);
}

}

MyTask 已经与存储库一起注入,因为您可以在 XML 映射中对其进行配置。我每天都使用这两种方法,但我倾向于第二种方法,因为它更容易阅读,并且通过确保开发人员不做不容易测试的事情来帮助保持开发人员的诚实,坦率地说,对于不经意的观察者来说,它更清晰。


推荐