如何在Spring中抽象java.time.Clock以进行测试

时间:2017-05-02 13:19:14

标签: java spring junit jsr310

我对问题Time dependent unit tests

有疑问

假设我构建了包含服务接口及其实现的Spring应用程序

如果我想在测试中更改时钟,我将不得不“污染”生产代码并与之接口。 setClock方法如下:

public interface MyService {
    void heavyBusinessLogic();
    void setClock(Clock clock);
}

@Service
public class MyServiceImpl implements MyService {

    private Clock clock = Clock.systemDefaultZone();

    @Override
    public void heavyBusinessLogic() {
        if (LocalDate.now(clock)...) {
            ... 
        }
    }

    @Override
    public void setClock(Clock clock) {
        this.clock = clock;
    }
}   

在测试中,我可以调用,例如:

service.setClock(Clock.fixed(Instant.parse("2017-02-14T01:22:00Z"), ZoneOffset.UTC));

如何在Spring中消除这种贯穿各领域的关注?

我想坚持使用java.time.Clock(我不想使用Joda)

2 个答案:

答案 0 :(得分:11)

就个人而言,我只想在构造函数中添加时钟......

public MyServiceImpl(Clock clock) {
  this.clock = clock;
}

...也许添加一个不错的默认构造函数......

public MyServiceImpl() {
  this(Clock.systemDefaultZone());
}

这样您就可以通过spring获取默认值并手动创建自定义时钟版本,例如在测试中。

当然,您也可以放弃默认构造函数,只需在生产配置中添加Clock bean,例如这样......

@Bean
public Clock clock() {
 return Clock.systemDefaultZone();
}

...允许您在测试配置中使用模拟Clock作为bean,通过构造函数注入自动允许Spring @Autowire

答案 1 :(得分:0)

如何为时钟建模的好方法是使用ThreadLocal,尤其是当您在JPA实体中需要时钟时,例如:

@Entity
public class MyEntity {
   public static final ThreadLocal<Clock> CLOCK =
            ThreadLocal.withInitial(Clock::systemDefaultZone);
}