以前的博客中已經介紹過dubbo服務的provider-consumer的實現,這裏實現一個內部服務由dubbo提供,而外部服務則用rest提供。實現的原理還須要進一步研究源代碼,但首先,咱們要知道how。本篇將展現第一種簡單配置,即將dubbo-provider做爲依賴,使用rest-provider啓動並提供服務。工程結構如圖:java
其中,dubbo-provider的結構在以前的博客中已有說明,它的重要配置文件依次以下:spring
dubbo.properties:json
dubbo.spring.config=classpath*:Spring/*.xml
Provider.xmltomcat
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:dubbo="http://code.alibabatech.com/schema/dubbo" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://code.alibabatech.com/schema/dubbo http://code.alibabatech.com/schema/dubbo/dubbo.xsd "> <!-- 提供方應用信息,用於計算依賴關係 --> <dubbo:application name="dubboProvider" /> <!-- 使用multicast廣播註冊中心暴露服務地址 <dubbo:registry address="multicast://224.5.6.7:1234" /> --> <!-- 使用zookeeper註冊中心暴露服務地址 --> <dubbo:registry address="zookeeper://127.0.0.1:2181" /> <!-- 用dubbo協議在20880端口暴露服務 --> <dubbo:protocol name="dubbo" port="20880" /> <!-- 具體的實現bean --> <bean id="ProviderServiceImpl" class="com.changjiang.test.dubboProvider.ProviderServiceImpl" /> <!-- 聲明須要暴露的服務接口 --> <dubbo:service interface="com.changjiang.test.dubboProvider.ProviderService" ref="ProviderServiceImpl" protocol="dubbo" version="dubboTest1.0"/> </beans>
接口以及實現:架構
package com.changjiang.test.dubboProvider; public class ProviderServiceImpl implements ProviderService { @Override public String provide() { System.out.println("== provide service =="); return "success"; } }
然後,咱們將其做爲一個依賴,放入dubboRest中:app
<dependency> <groupId>com.changjiang.test</groupId> <artifactId>dubboProvider</artifactId> <version>0.0.1-SNAPSHOT</version> </dependency>
這樣,咱們在rest服務的具體實現中就能夠調用該服務了。ide
dubboRest結構:ui
其中的dubbo.properties與上相同,Provider.xml:spa
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:dubbo="http://code.alibabatech.com/schema/dubbo" xmlns="http://www.springframework.org/schema/beans" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://code.alibabatech.com/schema/dubbo http://code.alibabatech.com/schema/dubbo/dubbo.xsd"> <!-- 提供方應用信息,用於計算依賴關係 --> <!-- 使用multicast廣播註冊中心暴露服務地址 --> <dubbo:registry address="zookeeper://127.0.0.1:2181" /> <dubbo:annotation package="com.changjiang.test.dubboRest" /> <dubbo:protocol path="DubboRest" name="rest" port="20660" server="tomcat" accepts="5000" threads="1000" keepalive="false" /> </beans>
這裏要注意的是,既然將dubboProvider做爲依賴引入,那麼就不能再指定一些重複的信息,好比dubbo-application,或者是發佈服務的端口號等。此處沒有設置dubboApplicationName,端口號改用20660,服務發佈的根路徑爲DubboRest。另外,引入applicationContext.xml是爲了加載文件,以及引用bean比較方便:3d
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xmlns:tx="http://www.springframework.org/schema/tx" xmlns="http://www.springframework.org/schema/beans" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:mongo="http://www.springframework.org/schema/data/mongo" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd http://www.springframework.org/schema/data/mongo http://www.springframework.org/schema/data/mongo/spring-mongo-1.0.xsd" default-lazy-init="true"> <description>Spring公共配置</description> <!-- 該 BeanPostProcessor 將自動對標註 @Autowired 的 Bean 進行注入 --> <context:annotation-config/> <!-- 使用annotation 自動註冊bean, 並保證@Required、@Autowired的屬性被注入 --> <context:component-scan base-package="com.changjiang.test.dubboRest"/> </beans>
不管是dubbo協議仍是rest協議服務,在dubbo架構中都必須使用接口實現,因此咱們看看接口以及它的實現:
package com.changjiang.test.dubboRest; import javax.ws.rs.Consumes; import javax.ws.rs.GET; import javax.ws.rs.Path; import javax.ws.rs.Produces; import javax.ws.rs.core.MediaType; import org.springframework.beans.factory.annotation.Autowired; import com.alibaba.dubbo.config.annotation.Service; import com.changjiang.test.dubboProvider.ProviderService; @Service(protocol="rest") @Path("/restService") @Consumes({ MediaType.APPLICATION_JSON }) @Produces({ MediaType.APPLICATION_JSON }) public class RestServiceImpl implements RestService{ @Autowired private ProviderService providerService; @Path("/getMsg") @GET public String rest() { String str = providerService.provide(); System.out.println(str); return str; } }
這裏@Service(protocol="rest")表示使用rest協議,然後咱們能夠根據下面的路徑和請求方式獲取到發佈出來的rest服務。
可能你已經注意到這裏用到了dubboFilter。對於Java Web應用,Spring的攔截器能夠攔截Web接口的調用;而對於dubbo接口,Spring的攔截器就無論用了。因此,須要使用相關的攔截和處理,須要用到dubboFilter:
/dubboRest/src/main/resources/META-INF/dubbo/com.alibaba.dubbo.rpc.Filter:
restValidateFilter=com.changjiang.test.dubboRest.interceptor.RestValidateFilter
實現:
package com.changjiang.test.dubboRest.interceptor; import javax.ws.rs.core.Response; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import com.alibaba.dubbo.rpc.Filter; import com.alibaba.dubbo.rpc.Invocation; import com.alibaba.dubbo.rpc.Invoker; import com.alibaba.dubbo.rpc.Result; import com.alibaba.dubbo.rpc.RpcException; import com.alibaba.dubbo.rpc.RpcResult; import com.alibaba.fastjson.JSONObject; public class RestValidateFilter implements Filter { Logger logger= LoggerFactory.getLogger(RestValidateFilter.class); @Override public Result invoke(Invoker<?> invoker, Invocation invocation) throws RpcException { return getRpcResult(invoker,invocation,null); } public Result getRpcResult(Invoker<?> invoker,Invocation invocation,JSONObject result){ if(invoker==null){ return new RpcResult(Response.status(401).build()); }else{ return invoker.invoke(invocation); } } }
配置都齊全了,咱們能夠在主類中使用dubbo自帶的Main啓動:
package com.changjiang.test.dubboRest; /** * Hello world! * */ public class App { public static void main(String[] args) { String[] ars = {}; com.alibaba.dubbo.container.Main.main(ars); } }
能夠在dubbo-admin中觀察到,此時咱們發佈了2個服務,一個是dubbo風格的,另外一個是rest風格的,而在此工程的內部,咱們已經調用到了該dubbo服務:
根據服務的路徑,咱們能夠用HttpGet請求進行訪問,查看結果: