Spring整合RESTEasy两种思路
本文的目的 是:更好的处理Spring Bean和RESTEasy Resource之间的关系
思路有两种:
- RESTEasy自己管理Resource的生命周期,在Resource生成时将需要的Spring bean手动注入进来
- 依靠Spring容器来管理Resource,将Spring的Bean当做Resource发布出来
下面比较详细的解释下:
(一)RESTEasy自己管理Resource的生命周期
具体思路是Resource层代码不使用Spring管理,在Resource的构造方法中手动注入需要的Spring Bean(比如Service方法等),上代码
@Path("book")
public class BookResource{
private BookService bookService;
public BookResource(){
BookService bookSerivce = (BookService)YOUR_SPRING_CONTEXT.getBean("bookService");
this.bookService = bookService;
}
@Path("{bookId:[0-9]*}")
@Get
@Produces({MediaType.APPLICATION_JSON,MediaType.APPLICATION_XML})
@Consumes({MediaType.APPLICATION_JSON,MediaType.APPLICATION_XML})
public Book getBookById(@PathParam("bookId")final Integer bookId){
final Book book = this.bookService.getBookById(bookId);
return book;
}
}
RESTEasy是在接收到请求的时候通过反射利用构造方法生成Resource实例的,所以这里我们重写了Resource的构造方法。在初始化的时候手动去SpringContext中取到需要的bean,将其注入进来,这样就实现了在Resource中调用Service方法。
(二)依靠Spring容器来管理Resource
先看看下面是 chapter45.spring integration中web.xml中需要的配置:
<web-app>
<listener>
<listener-class>
org.jboss.resteasy.plugins.server.servlet.ResteasyBootstrap
</listener-class>
</listener>
<!-- ** INSERT YOUR LISTENERS HERE!!!! -->
<!-- resteasy自定义的spring容器 -->
<listener>
<listener-class>
org.jboss.resteasy.plugins.spring.SpringContextLoaderListener
</listener-class>
</listener>
<servlet>
<servlet-name>Resteasy</servlet-name>
<servlet-class>
org.jboss.resteasy.plugins.server.servlet.HttpServletDispatcher
</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>Resteasy</servlet-name>
<url-pattern>/Resteasy/*</url-pattern>
</servlet-mapping>
</web-app>
可以看出上面配置了resteasy自定义的SpringContextLoaderListener。
这样做的局限是:由于SpringContext只能存在一个,所以下面的spring容器不可以和上面的RESTEasy自定义spring容器共存。
如果你的项目没有自定义Spring容器,而使用的是原始的Spring容器的话,考虑直接用上面代码中的 org.jboss.resteasy.plugins.spring.SpringContextLoaderListener替换掉你原来的 org.springframework.web.context.ContextLoaderListener来解决,就不用往下看了。
<!-- 当前项目自定义的spring容器 -->
<listener>
<listener-class>
custom.SpringContextLoaderListener
</listener-class>
</listener>
那么解决思路就很明确了。
将RESTEasy的spring实现整合到项目自定义的spring实现中
通过分析 resteasy-spring源码包发现,RESTEasy的自定义SpringContextLoader代码入口在 org.jboss.resteasy.plugins.spring.SpringContextLoader.java中,此类代码如下:
public class SpringContextLoader extends ContextLoader
{
private SpringContextLoaderSupport springContextLoaderSupport = new SpringContextLoaderSupport();
protected void customizeContext(ServletContext servletContext, ConfigurableWebApplicationContext configurableWebApplicationContext)
{
super.customizeContext(servletContext, configurableWebApplicationContext);
this.springContextLoaderSupport.customizeContext(servletContext,configurableWebApplicationContext);
}
}
到这里就简单了,我们在自定义的CustomSpringContextLoader中加入上面的代码,OK
public class CustomSpringContextLoader extends ContextLoader
{
<!-- YOUR CODE!!! -->
private SpringContextLoaderSupport springContextLoaderSupport = new SpringContextLoaderSupport();
protected void customizeContext(ServletContext servletContext, ConfigurableWebApplicationContext configurableWebApplicationContext)
{
super.customizeContext(servletContext, configurableWebApplicationContext);
this.springContextLoaderSupport.customizeContext(servletContext,configurableWebApplicationContext);
}
}
接下来要做的就是通过注解或者配置的方式将BookResource加载到Spring容器。
@Component
@Scope("request")
@Path("book")
public class BookResource{
private BookService bookService;
public BookResource(){
BookService bookSerivce = (BookService)YOUR_SPRING_CONTEXT.getBean("bookService");
this.bookService = bookService;
}
@Path("{bookId:[0-9]*}")
@Get
@Produces({MediaType.APPLICATION_JSON,MediaType.APPLICATION_XML})
@Consumes({MediaType.APPLICATION_JSON,MediaType.APPLICATION_XML})
public Book getBookById(@PathParam("bookId")final Integer bookId){
final Book book = this.bookService.getBookById(bookId);
return book;
}
}
我这里通过 @Component来加载的。
是不是看到了 @Scope注解,为什么要这样呢?
已有 0 人发表留言,猛击->> 这里<<-参与讨论
ITeye推荐