3 回答
TA贡献1813条经验 获得超2个赞
使用多种实现
一种可能性是创建MailService具有两个实现的接口。通过将这些实现与@Profile注释相结合,您可以根据您使用的配置文件正确注入一种或另一种实现。例如:
public interface MailService {
void send(String subject, String text, String... emails);
}
然后您可以将@Profile("mail")注释添加到一个实现中:
@Service
@Profile("mail") // Only create this bean when the 'mail' profile is used
public class JavaMailServiceImpl implements MailService {
@Autowired
private JavaMailSender mailSender;
public void send(String subject, String text, String... emails) {
// MailMessage object configuration
mailSender.send(mail);
}
}
此外,您可以在邮件配置文件未激活时添加另一个实现:
@Service
@Profile("!mail") // By adding the exclamination mark, this implementation will be used when the mail profile isn't active
public class NoopMailServiceImpl implements MailService {
private final Logger logger = LoggerFactory.getLogger(getClass());
@Override
public void send(String subject, String text, String... emails) {
logger.debug("Dummy implementation, no e-mail is being sent");
}
}
如果您现在自动连接MailService到控制器中,它将根据您使用的配置文件正确实现正确的实现。
使用方面
另一种替代方法是使用方面将发送邮件与订单创建逻辑完全解耦。
这允许您完全删除控制器MailService的依赖关系MyRestController:
@RestController
public class MyRestController {
@PostMapping("/orders")
public void createOrders(@RequestBody List<Order> orders) {
// manipulations with orders
}
}
相反,您可以将mailService逻辑添加到单独的方面:
@Aspect
@Component
@Profile("mail")
public class OrderNotificationAspect {
private final MailService mailService;
public OrderNotificationAspect(MailService mailService) {
this.mailService = mailService;
}
@AfterReturning("execution(* com.example.MyRestController.createOrders(..))")
public void sendNotification() {
mailService.send("Order notification", "New orders", "1@mail.com");
}
}
像以前一样,我们使用@Profile注释仅在邮件配置文件处于活动状态时注册方面 bean。然而,由于我们不再直接将控制器绑定到MailService我们不再需要“虚拟实现”。
要在 Spring boot 中使用切面,您应该添加spring-boot-starter-aop依赖项。
以编程方式检查配置文件
另一种选择是在代码中编写检查,例如:
@RestController
public class MyRestController {
@Autowired
private MailService mailService;
@Autowired
private Environment environment;
@PostMapping("/orders")
public void createOrders(@RequestBody List<Order> orders) {
// manipulations with orders
if (environment.acceptsProfiles(Profiles.of("mail"))) {
mailService.send("Order notification", "New orders", "1@mail.com");
}
}
}
根据您希望此逻辑的细粒度程度,您可以将其放入控制器或服务中(如果您想禁用为特定配置文件发送的所有电子邮件)。
您选择哪个选项取决于您的需求。使用多种实现是禁用所有邮件发送的最简单方法。
但是,如果您只想禁用某些邮件的发送,则必须选择其他方式。使用方面的缺点是它有时会使流程更难以遵循,尽管 IDE 对方面有很好的支持。
TA贡献1824条经验 获得超8个赞
只需为需要它的配置文件提供一个虚拟实现
@Bean
@Profile("prod")
public JavaMailSender getRealJavaMailSender() {
JavaMailSenderImpl mailSender = new JavaMailSenderImpl();
mailSender.setHost("smtp.gmail.com");
mailSender.setPort(587);
mailSender.setUsername("my.gmail@gmail.com");
mailSender.setPassword("password");
Properties props = mailSender.getJavaMailProperties();
props.put("mail.transport.protocol", "smtp");
props.put("mail.smtp.auth", "true");
props.put("mail.smtp.starttls.enable", "true");
props.put("mail.debug", "true");
return mailSender;
}
@Bean
@Profile("test")
public JavaMailSender getDummyJavaMailSender() {
return new JavaMailSender() {
... dummy method implementations ...
};
}
TA贡献1818条经验 获得超11个赞
sendEmails=false我建议向 application-[profile-name].properties 文件添加一个变量(例如),每个配置文件具有不同的设置。
然后,在您的代码中,您可以使用此属性来决定是否应发送电子邮件。
@Service
public class MailService {
@Value("sendEmails}")
private boolean sendEmails;
@Autowired
private JavaMailSender mailSender;
public void send(String subject, String text, String... emails) {
if (sendEmails) {
// MailMessage object configuration
mailSender.send(mail);
}
}
}
这种方法使您能够为每个配置文件使用不同的(且易于更改的)设置。
添加回答
举报