admin管理员组文章数量:1343915
I have this application that works fine for a single job:
interface Job {
void foo();
}
class JobA implements Job {
void foo() {...}
}
class JobB implements Job {
void foo() {...}
}
class JobExecutor {
Job job;
// autowired constructor
public JobExecutor(Job job) {this.job = job;}
}
@Configuration
public class MyConfiguration {
@ConditionalOnProperty(name = "job.name", havingValue = "jobA")
@Bean
public Job jobA() {
return new JobA();
}
@ConditionalOnProperty(name = "job.name", havingValue = "jobB")
@Bean
public Job jobB() {
return new JobB();
}
@Bean
public JobExecutor jobExecutor(Job job) {
return new JobExecutor(job);
}
}
config:
job.name = jobA # or jobB
But now I want to extend the application to take 2 jobs so the config is updated like this:
job1.name = jobA # or jobB or jobC
job2.name = jobA # or jobB or jobC
then the config is as follows:
@Configuration
public class MyConfiguration {
@ConditionalOnExpression("#{'${job1.name}' == 'jobA' or '${job2.name}' == 'jobA'}")
@Bean
public Job jobA() {
return new JobA();
}
@ConditionalOnExpression("#{'${job1.name}' == 'jobB' or '${job2.name}' == 'jobB'}")
@Bean
public Job jobB() {
return new JobB();
}
@ConditionalOnExpression("#{'${job1.name}' == 'jobC' or '${job2.name}' == 'jobC'}")
@Bean
public Job jobC() {
return new JobC();
}
}
then the Job classes are as follows (where every job is a subtype of Job
):
interface Job {
void foo();
}
class JobA implements Job {
void foo() {...}
}
class JobB implements Job {
void foo() {...}
}
class JobC implements Job {
void foo() {...}
}
class JobExecutor {
Job job1;
Job job2;
// autowired constructor -- spring complains about multiple beans.
public JobExecutor(Job job1, job2) { .... }
}
but the problem comes when spring tries to autowire the Job
classes because now there are two and it is also possible that the job1 and job2 are configured to the same Job eg JobA.. in this case it works fine because only one bean of the Job type is created
but how to use mutliple jobs ? I thought about using the @Qualifier
tag but this is not really possible because the job comes from config
I have this application that works fine for a single job:
interface Job {
void foo();
}
class JobA implements Job {
void foo() {...}
}
class JobB implements Job {
void foo() {...}
}
class JobExecutor {
Job job;
// autowired constructor
public JobExecutor(Job job) {this.job = job;}
}
@Configuration
public class MyConfiguration {
@ConditionalOnProperty(name = "job.name", havingValue = "jobA")
@Bean
public Job jobA() {
return new JobA();
}
@ConditionalOnProperty(name = "job.name", havingValue = "jobB")
@Bean
public Job jobB() {
return new JobB();
}
@Bean
public JobExecutor jobExecutor(Job job) {
return new JobExecutor(job);
}
}
config:
job.name = jobA # or jobB
But now I want to extend the application to take 2 jobs so the config is updated like this:
job1.name = jobA # or jobB or jobC
job2.name = jobA # or jobB or jobC
then the config is as follows:
@Configuration
public class MyConfiguration {
@ConditionalOnExpression("#{'${job1.name}' == 'jobA' or '${job2.name}' == 'jobA'}")
@Bean
public Job jobA() {
return new JobA();
}
@ConditionalOnExpression("#{'${job1.name}' == 'jobB' or '${job2.name}' == 'jobB'}")
@Bean
public Job jobB() {
return new JobB();
}
@ConditionalOnExpression("#{'${job1.name}' == 'jobC' or '${job2.name}' == 'jobC'}")
@Bean
public Job jobC() {
return new JobC();
}
}
then the Job classes are as follows (where every job is a subtype of Job
):
interface Job {
void foo();
}
class JobA implements Job {
void foo() {...}
}
class JobB implements Job {
void foo() {...}
}
class JobC implements Job {
void foo() {...}
}
class JobExecutor {
Job job1;
Job job2;
// autowired constructor -- spring complains about multiple beans.
public JobExecutor(Job job1, job2) { .... }
}
but the problem comes when spring tries to autowire the Job
classes because now there are two and it is also possible that the job1 and job2 are configured to the same Job eg JobA.. in this case it works fine because only one bean of the Job type is created
but how to use mutliple jobs ? I thought about using the @Qualifier
tag but this is not really possible because the job comes from config
1 Answer
Reset to default 0Use a Map<String, Job> injection pattern and dynamically select based on config values
Spring can inject all beans of type Job into a Map<String, Job>, where the bean name is the key.
This allows you to use configuration properties like job1.name and job2.name to look up the right bean at runtime, without relying on multiple conditional bean definitions.
1. Define jobs as regular beans with specific names
@Configuration
public class MyConfiguration {
@Bean("jobA")
public Job jobA() {
return new JobA();
}
@Bean("jobB")
public Job jobB() {
return new JobB();
}
@Bean("jobC")
public Job jobC() {
return new JobC();
}
@Bean
public JobExecutor jobExecutor(
Map<String, Job> jobMap,
@Value("${job1.name}") String job1Name,
@Value("${job2.name}") String job2Name) {
Job job1 = jobMap.get(job1Name);
Job job2 = jobMap.get(job2Name);
return new JobExecutor(job1, job2);
}
}
2. Your JobExecutor class
public class JobExecutor {
private final Job job1;
private final Job job2;
public JobExecutor(Job job1, Job job2) {
this.job1 = job1;
this.job2 = job2;
}
public void execute() {
job1.foo();
job2.foo();
}
}
3. Your application.properties
job1.name=jobA
job2.name=jobB
No need for @ConditionalOnExpression or @ConditionalOnProperty on every bean
本文标签: javaMultiple beans in contexthow to dynamically select based on configStack Overflow
版权声明:本文标题:java - Multiple beans in context - how to dynamically select based on config? - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743738558a2530463.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论