gpt4 book ai didi

java - 访问 Spring 上下文

转载 作者:行者123 更新时间:2023-12-04 04:59:03 27 4
gpt4 key购买 nike

让我对 Spring 有点困惑的东西。我知道它的一个主要功能是通过提供一种将模块化组件(类)连接在一起的方式来促进模块化和松散耦合,而无需了解其他模块的实现,只知道它们公开的契约(Contract)。

但是,在实践中,我有时会发现自己处于以下情况。

一个类必须获得 spring 应用程序上下文。假设您在 main();

因此,您在 main 中加载 spring,然后希望您的其余对象现在作为 bean 很好地注入(inject),并且可以在不紧密耦合的情况下进行交互。

但是,如果其中一个对象有一个每次调用时都需要创建一个新 bean 的方法呢?该对象现在需要对 spring 应用程序上下文的引用。但现在它不再是真正的 pojo,因为它与 spring 耦合?

我想在实践中,您创建了一个包含 Spring 应用程序上下文的全局对象,以便需要执行此操作的类轻松访问,但它仍然看起来有点困惑,并且与 Spring 的松散耦合哲学背道而驰?

这只是其中之一吗?它并不完美,但它仍然比让你的类紧密耦合要好得多?

编辑:如果你使用一个全局对象来保存你的 spring 应用程序上下文,你如何对访问它的类进行单元测试?您是否编写了全局持有者,以便可以使用测试上下文对其进行参数化。或者,如果您不使用全局对象,而是将 spring 上下文注入(inject)到需要它的类中,这意味着您需要通过代码保持注入(inject) spring 的链,否则您将到达一个点想要一个非 Spring 化的对象来创建一个新的 bean,但没有引用 Spring ?

最佳答案

这是工厂/服务如何提供帮助的人为示例。它使用 @Inject , 但原则适用于 @Autowired , ETC。

Spring docs for testing are here .

想象一下 IsSummerService ,它提供了一种方法来告诉您“现在”是否是夏天:

public interface IsItSummerService {
public boolean isItSummerNow();
}

一个简单的第一个 impl 可能如下所示。这个 impl 使用 new关键字来管理其 Date依赖和 @Inject管理夏季月份范围。 Date这里的依赖类似于你创建一个新的 bean。
import java.util.*;
import javax.inject.Inject;

public class IsItSummerServiceImpl1 implements IsItSummerService {
@Inject int startMonth = Calendar.JUNE;
@Inject int endMonth = Calendar.SEPTEMBER;
public boolean isItSummerNow() {
return DateUtils.isDateBetweenMonths(new Date(), startMonth, endMonth);
}
}

第二个 impl 可以使用 DI 注入(inject) NowService (或工厂)。此服务/工厂将取消使用 new ,并将创建新鲜 bean 的责任传递给服务/工厂。
import java.util.*;
import javax.inject.Inject;

public class IsItSummerServiceImpl2 implements IsItSummerService {
@Inject NowService nowService;
@Inject int startMonth = Calendar.JUNE;
@Inject int endMonth = Calendar.SEPTEMBER;
public boolean isItSummerNow() {
return DateUtils.isDateBetweenMonths(nowService.now(), startMonth, endMonth);
}
}

现在服务:
import java.util.Date;

public interface NowService {
public Date now();
}

那么现在每个测试有多难?无论您在哪里看到“假注入(inject)”,都可以通过 Spring 和测试上下文来实现。
IsItSummerServiceImpl2 的测试显然更好,更灵活。
import java.text.*;
import java.util.*;
import org.junit.*;

import static org.junit.Assert.*;

public class IsItSummerServiceTest {
@Test
public void testIsItSummerImpl1() {
IsItSummerService s = new IsItSummerServiceImpl1();
assertFalse(s.isItSummerNow());
// Test passes for May in the UK
// But what about when 'now' changes?
// We can't control the 'now' value effectively to test.
// assertFalse could pass/fail unpredictably.
// (Well, it actually *is* predictable when it passes/fails of course!)
}
@Test
public void testIsItSummerImpl2() {
// You would keep these tests separate, but for the sake of brevity.
IsItSummerService s = new IsItSummerServiceImpl2();

// Fake injection - set values directly. Use mocks/stubs/whatever.
((IsItSummerServiceImpl2)s).nowService = fakeNowService("01/01/2013");
assertFalse(s.isItSummerNow());

((IsItSummerServiceImpl2)s).nowService = fakeNowService("01/07/2013");
assertTrue(s.isItSummerNow());
}
@Test
public void testIsItSummerInSouthernHemisphere() {
// You would keep these tests separate, but for the sake of brevity.
IsItSummerService s = new IsItSummerServiceImpl2();

// Fake injection - set values directly. Use mocks/stubs/whatever.
((IsItSummerServiceImpl2)s).startMonth = Calendar.DECEMBER;
((IsItSummerServiceImpl2)s).endMonth = Calendar.MARCH;

((IsItSummerServiceImpl2)s).nowService = fakeNowService("01/11/2013");
assertFalse(s.isItSummerNow());

((IsItSummerServiceImpl2)s).nowService = fakeNowService("01/12/2013");
assertTrue(s.isItSummerNow());

((IsItSummerServiceImpl2)s).nowService = fakeNowService("01/03/2013");
assertTrue(s.isItSummerNow());

((IsItSummerServiceImpl2)s).nowService = fakeNowService("01/04/2013");
assertFalse(s.isItSummerNow());
}

static class FakeNowService implements NowService {
Date now;
public FakeNowService(Date now) {
this.now = now;
}
public Date now() {
return now;
}
}

Date parseDate(String dateStr) {
try {
// UK date format
return new SimpleDateFormat("dd/MM/yyyy").parse(dateStr);
} catch (ParseException e) {
throw new RuntimeException(e);
}
}

NowService fakeNowService(String dateStr) {
return new FakeNowService(parseDate(dateStr));
}
}

因此,通过将对象创建从 new 转移到您的问题上。关键字我们使应用程序更具可测试性。

使用 new如果您有可能希望在以后更改对象创建过程,则不希望这样做。一次 new已经用过了,差不多就是这样。但是有了工厂,您可以随心所欲地改变创建过程。

关于java - 访问 Spring 上下文,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16372396/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com