gpt4 book ai didi

java - 使用 Play! 编写一个简单的 Spring 示例框架

转载 作者:搜寻专家 更新时间:2023-11-01 03:44:16 25 4
gpt4 key购买 nike

我正在尝试设置 Spring 1.0.1 模块 以与 Play! 1.2.1.

首先,我将 Spring 模块添加到 dependencies.yml 中:

- play -> spring 1.0.1

然后我根据 documentation 添加了以下内容在 application.conf 中:

play.spring.component-scan=true

然后我运行以下命令来下载依赖项并更新我的 IntelliJ 配置:

play deps --sync
play idealize

现在,当我返回 IntelliJ 时,我可以看到 spring-1.0.1 已添加到我的项目模块列表中,看起来是正确的。但是,如果我查看项目的依赖项,它不会在 spring-1.0.1/lib 目录中找到 jar。如果我尝试用@Component 注释我的一个类,它找不到它。但是,如果我手动添加 spring-1.0.1/lib 作为项目依赖项,它会发现 @Component 就好了。我是否应该运行其他程序来让我的 IntelliJ 项目识别模块中的 spring 库?

我问题的第二部分涉及连接一个简单示例并为其编写测试。

假设我有一个这样的 Job 类:

@Component
@Every("10min")
public class MyJob extends Job {
private @Inject Printer printer;

@Override
public void doJob() throws Exception {
printer.print();
}
}

打印机类如下所示:

@Component
public class Printer {
public void print() {
System.out.println("foo");
}
}

首先,注释看起来正确吗?其次,我将如何编写一个调用 MyJob.doJob() 的测试,但注入(inject)一个不同的打印机来打印“bar”而不是“foo”?

更新: 正如 Aaron 所提到的,我将 @Inject 更改为 @Autowired。在我的测试课上,我有这样的事情:

@Test
public void testSomething() {
MyJob job = play.modules.spring.Spring.getBeanOfType(MyJob.class);
job.doJob();
...
}

它在 getBeanOfType 语句上失败,带有以下内容:

play.modules.spring.SpringException
at play.modules.spring.Spring.getBean(Spring.java:11)
at com.testing.JobTest.testSomething(MyJob.java:20)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:44)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:41)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:20)
at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:28)
at play.test.PlayJUnitRunner$StartPlay$1$1$1.execute(PlayJUnitRunner.java:73)
at play.Invoker$Invocation.run(Invoker.java:265)
at play.Invoker.invokeInThread(Invoker.java:67)
at play.test.PlayJUnitRunner$StartPlay$1$1.evaluate(PlayJUnitRunner.java:68)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:76)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:50)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:193)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:52)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:191)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:42)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:184)
at org.junit.runners.ParentRunner.run(ParentRunner.java:236)
at play.test.PlayJUnitRunner.run(PlayJUnitRunner.java:48)
at org.junit.runner.JUnitCore.run(JUnitCore.java:157)
at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:60)

更新 2:好吧,我现在已经更进一步了。似乎即使我正在进行组件扫描,Spring 模块仍然需要 conf/目录中的 application-context.xml(即使它没有定义 bean)。

现在我只需要弄清楚如何注入(inject)不同版本的 Printer 对象,它会输出“bar”而不是“foo”。

更新 3:我已将 @Autowired 替换为 @Resource(name="printer"),并且已使用 @Component("printer") 注释我的生产打印机并将其重命名为 RealPrinter 和提供了一个接口(interface):

@Component("printer")
public class RealPrinter implements Printer {
public void print() {
System.out.println("foo");
}
}

如果我只有上述打印机,注入(inject)工作正常。现在,如果我创建一个 TestPrinter,我需要 Play 的 Spring 模块来意识到我想在执行测试类时使用 TestPrinter 而不是 RealPrinter。

@Component("printer")
public class TestPrinter implements Printer {
public void print() {
System.out.println("bar");
}
}

但是,因为我有两个组件限定名称为“打印机”,Play 的 Spring 模块抛出以下异常:

play.exceptions.UnexpectedException: Unexpected Error
at play.Play.start(Play.java:491)
at play.test.PlayJUnitRunner.<init>(PlayJUnitRunner.java:31)
at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:39)
at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:27)
at java.lang.reflect.Constructor.newInstance(Constructor.java:513)
at org.junit.internal.builders.AnnotatedBuilder.buildRunner(AnnotatedBuilder.java:31)
at org.junit.internal.builders.AnnotatedBuilder.runnerForClass(AnnotatedBuilder.java:24)
at org.junit.runners.model.RunnerBuilder.safeRunnerForClass(RunnerBuilder.java:57)
at org.junit.internal.builders.AllDefaultPossibilitiesBuilder.runnerForClass(AllDefaultPossibilitiesBuilder.java:29)
at org.junit.runners.model.RunnerBuilder.safeRunnerForClass(RunnerBuilder.java:57)
at org.junit.internal.requests.ClassRequest.getRunner(ClassRequest.java:24)
at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:66)
at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:192)
at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:60)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at com.intellij.rt.execution.application.AppMain.main(AppMain.java:115)
Caused by: java.lang.IllegalStateException: Annotation-specified bean name 'printer' for bean class [com.testing.job.RealPrinter] conflicts with existing, non-compatible bean definition of same name and class [com.testing.job.TestPrinter]
at org.springframework.context.annotation.ClassPathBeanDefinitionScanner.checkCandidate(ClassPathBeanDefinitionScanner.java:267)
at org.springframework.context.annotation.ClassPathBeanDefinitionScanner.doScan(ClassPathBeanDefinitionScanner.java:208)
at org.springframework.context.annotation.ClassPathBeanDefinitionScanner.scan(ClassPathBeanDefinitionScanner.java:180)
at play.modules.spring.SpringPlugin.onApplicationStart(SpringPlugin.java:98)
at play.plugins.PluginCollection.onApplicationStart(PluginCollection.java:408)
at play.Play.start(Play.java:463)
... 19 more

如您所见,异常的原因是现在有两个同名的打印机。

最佳答案

好的,我想出了一个适合我的解决方案。因此,为了编写注入(inject)模拟/测试版本的 bean 的测试,我执行了以下操作:

@Test
public void testSomething() {
GenericApplicationContext context = SpringPlugin.applicationContext;
GenericBeanDefinition definition = new GenericBeanDefinition();
definition.setBeanClass(TestPrinter.class);
definition.setAutowireCandidate(true);
context.registerBeanDefinition("printer", definition);

MyJob job = new MyJob();
job.doJob();
}

然后我不得不将 MyJob 更改为它需要的东西的根“注入(inject)器”:

public class MyJob extends Job {
public void doJob() throws Exception {
Printer printer = Spring.getBeanOfType(Printer.class);
printer.print();
}
}

TestPrinter 不再需要使用@Component 注释,因为测试方法以编程方式注册测试打印机 bean,从而覆盖生产打印机 bean。

因此执行 testSomething() 方法将导致打印出“bar”。如果我删除 registerBeanDefinition 内容,将导致打印出生产“foo”消息。

我不确定这是否是使用 Play 的 Spring 模块注入(inject)测试内容的最简洁方法,但至少我现在可以做我想做的事了。

关于java - 使用 Play! 编写一个简单的 Spring 示例框架,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6183954/

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