使用SpringFramework+Restlet实现rest服务

2022-10-27 09:51:54 浏览数 (2)

什么是REST

REST 全称是 Representational State Transfer(表述性状态转移),它是 Roy Fielding 博士在 2000 年写的一篇关于软件架构风格的论文。许多知名互联网公司开始采用这种轻量级 Web 服务,大家习惯将其称为 RESTful Web Services,或简称 REST 服务。

REST 本质上是使用 URL 来访问资源的一种方式。总所周知,URL 就是我们平常使用的请求地址了,其中包括两部分:请求方式 与 请求路径,比较常见的请求方式是 GET 与 POST,但在 REST 中又提出了几种其它类型的请求方式,汇总起来有六种:GET、POST、PUT、DELETE、HEAD、OPTIONS。尤其是前四种,正好与 CRUD(增删改查)四种操作相对应:GET(查)、POST(增)、PUT(改)、DELETE(删)。

实际上,REST 是一个“无状态”的架构模式,因为在任何时候都可以由客户端发出请求到服务端,最终返回自己想要的数据。也就是说,服务端将内部资源发布 REST 服务,客户端通过 URL 来访问这些资源,这不就是 SOA 所提倡的“面向服务”的思想吗?所以,REST 也被人们看做是一种轻量级的 SOA 实现技术,因此在企业级应用与互联网应用中都得到了广泛使用。

在 Java 的世界里,有一个名为 JAX-RS 的规范,它就是用来实现 REST 服务的。目前有许多框架已经实现了该规范,比如restlet、cxf。

restlet可以单独使用,也可以与springframework继承一起使用,下面讲解第二种。

使用 Spring restlet 发布 REST 服务

添加maven依赖

代码语言:javascript复制
<dependencies>
	<!-- Spring -->
	<dependency>
		<groupId>org.springframework</groupId>
		<artifactId>spring-web</artifactId>
		<version>${spring.version}</version>
	</dependency>
	<!-- restlet -->
	<dependency>
		<groupId>org.restlet.jee</groupId>
		<artifactId>org.restlet</artifactId>
	</dependency>
	<dependency>
		<groupId>org.restlet.jee</groupId>
		<artifactId>org.restlet.ext.jackson</artifactId>
		<exclusions>
			<exclusion>
				<groupId>org.codehaus.woodstox</groupId>
				<artifactId>woodstox-core-asl</artifactId>
			</exclusion>
		</exclusions>
	</dependency>
	<dependency>
		<groupId>org.restlet.jee</groupId>
		<artifactId>org.restlet.ext.servlet</artifactId>
	</dependency>
	<dependency>
		<groupId>org.restlet.jee</groupId>
		<artifactId>org.restlet.ext.spring</artifactId>
	</dependency>
</dependencies>

这里仅依赖 Spring Web 模块(无需 MVC 模块)。

配置web.xml

代码语言:javascript复制
<context-param>
	<param-name>contextConfigLocation</param-name>
	<param-value>
			  classpath*:META-INF/spring/*.service.xml
			  /WEB-INF/classes/*.datasource.xml 
			  /WEB-INF/classes/*.service.xml
			  classpath*:conf/*.service.xml
			  classpath*:*.service.xml	
	</param-value>
</context-param>

......

<listener>
	<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> 
</listener>
<servlet>
	<servlet-name>restlet</servlet-name>
	<servlet-class>org.restlet.ext.spring.RestletFrameworkServlet</servlet-class>
	<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
	<servlet-name>restlet</servlet-name>
	<url-pattern>/*</url-pattern>
</servlet-mapping>

使用 Spring 提供的 ContextLoaderListener 去加载所有的配置文件*.xml;使用 restlet 提供的 RestletFrameworkServlet 去处理前缀为所有的 REST 请求。

定义Resource资源类

代码语言:javascript复制
@Controller("backupPolicyCreateAndQuery")
@Scope("prototype")
public class BackupPolicyCreateAndQuery extends ServerResource
{
	@Post
    public BackupPolicyCreateResponse createPolicy(BackupPolicyCreateRequest request)
    {......}
}

创建resource代码类,要继承org.restlet.resource.ServerResource,用@Get和@Post注解来什么类型的请求

配置spring文件

下面只举一个示例,spring配置相信大家都比较熟悉了,就不多说

代码语言:javascript复制
<context:component-scan base-package="com.huawei.hwclouds.autobackup"/>
<context:annotation-config />
	
<bean id="scheduleConfigs"
	class="org.springframework.beans.factory.config.PropertiesFactoryBean">
	<property name="locations">
		<list>
			<value>classpath:service.config.properties</value>
		</list>
	</property>
</bean> 

<bean id="scheduler"
	class="org.springframework.scheduling.quartz.SchedulerFactoryBean" autowire="no">
	<property name="configLocation" value="classpath:quartz.properties" />
</bean>

......

将接口的实现类发布为SpringBean

有两种方式:一是使用spring配置文件;一是使用注解。这里采用第二种,如上面定义Resource资源类所示:

创建restlet-servlet.xml

在web.xml同目录下创建{name}-servlet.xml,其中name为web.xml中定义的servlet名称

代码语言:javascript复制
<bean name="root" class="org.restlet.ext.spring.SpringRouter">
	<property name="attachments">
		<map>
			<!-- 此处可以按业务模块划分url v1 or v2 -->
			<entry key="/v1/{tenant}" value-ref="v1Route" />
			<entry key="/v2/{tenant}" value-ref="v2Route" />
			
			<entry key="/server_status">
				<bean class="org.restlet.ext.spring.SpringFinder">
					<lookup-method name="create" bean="autoBackupStatusResource" />
				</bean>
			</entry>
		</map>
	</property>
</bean>

<bean id="v1Route" class="org.restlet.ext.spring.SpringRouter">
	<property name="attachments">
		<map>
			<entry key="/backuppolicy">
				<bean class="org.restlet.ext.spring.SpringFinder">
					<lookup-method name="create"
						bean="backupPolicyCreateAndQuery" />
				</bean>
			</entry>
		</map>
	</property>
</bean>

这个文件主要是配置url的路径对应的资源bean,可以看到,这里的bean也是可以引用的。

0 人点赞