前言java
週一入職的新公司,到了公司第一件事天然是要熟悉新公司使用的各類技術,搭建本地的環境。算法
熟悉新公司技術的過程當中,首先就是Maven,這個前面已經寫過文章了,而後就是Dubbo----公司的服務都是經過Dubbo來治理的。其實以前我就對SOA、RPC等分佈式服務的概念有所瞭解,Dubbo也多多少少知道一些,不過能親身使用那是再好不過的了。spring
此文就對個人Dubbo學習作一個小的總結,基本我總結的這些內容對於入門應該是足夠了。關於Dubbo的更詳細信息,阿里有專門出手冊,手冊地址爲http://dubbo.io/User+Guide-zh.htm,本文也只是對手冊的重點進行一些提煉而已(看過就知道了,這篇手冊實在是太龐大了),要進一步地瞭解Dubbo的朋友們仍是應當去看Dubbo的官方手冊。json
Dubbo是什麼緩存
Dubbo是一個分佈式服務框架,致力於提供高性能和透明化的RPC遠程服務調用方案,以及SOA服務治理方案。其核心部分包括:網絡
Dubbo能作什麼架構
Dubbo功能強大,總結下來,它大體能夠作如下幾件事:併發
Dubbo架構圖app
這是最重要的,理解Dubbo的架構圖是理解Dubbo的第一步,我從Dubbo官網手冊上截了一下Dubbo架構圖:負載均衡
在接下來的講解以前,說明一個概念:所謂SOA也好,分佈式服務框架也好,不是服務消費者從中間件(通常都是Zookeeper)上去拿數據,而是服務消費者從中間件上拿到可用的服務生產者的集羣地址,再從集羣地址中選出一個進行直連。
接下來認識一下圖中的結點:
圖中已經有了調用步驟了,接着對步驟進行說明:
Dubbo用法
Dubbo採用全Spring配置方式,透明化接入應用,對應用沒有任何API侵入(這句話的意思是應用程序不會、不須要手動調用Dubbo的任何類和任何接口),只需用Spring加載Dubbo的配置便可(意思是對Dubbo的使用只須要寫Spring配置文件或註解),Dubbo基於Spring的Schema擴展進行加載。
若是不想使用Spring配置,而但願經過API的方式進行調用,能夠本身看一下官方手冊Dubbo API配置,可是,這種作法十分不推薦。
下面簡單說明一下Dubbo的用法,首先要定義服務生產者的接口及其實現,那麼定義一個接口(該接口須要單獨打包,在生產者和消費者之間共享):
package com.xrq.dubbo.demo; public interface DemoService { String sayHello(String name); }
在服務生產者處實現接口(對服務消費者隱藏接口實現細節):
package com.xrq.dubbo.demo.provider; import com.xrq.dubbo.demo.DemoService; public class DemoServiceImpl implements DemoService { public String sayHello(String name) { return "Hello " + name; } }
寫一個provider.xml,在服務生產者使用Spring暴露服務:
<?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="hello-world-app" /> <!-- 使用multicast廣播註冊中心暴露服務地址 --> <dubbo:registry address="multicast://224.5.6.7:1234" /> <!-- 用dubbo協議在20880端口暴露服務 --> <dubbo:protocol name="dubbo" port="20880" /> <!-- 聲明須要暴露的服務接口 --> <dubbo:service interface="com.xrq.dubbo.demo.DemoService" ref="demoService" /> <!-- 和本地bean同樣實現服務 --> <bean id="demoService" class="com.xrq.dubbo.demo.provider.DemoServiceImpl" /> </beans>
在服務消費者處寫一個consumer.xml引用遠程服務:
<?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="consumer-of-helloworld-app" /> <!-- 使用multicast廣播註冊中心暴露發現服務地址 --> <dubbo:registry address="multicast://224.5.6.7:1234" /> <!-- 生成遠程服務代理,能夠和本地bean同樣使用demoService --> <dubbo:reference id="demoService" interface="com.alibaba.dubbo.demo.DemoService" /> </beans>
服務消費者處只要經過Spring拿到demoService,便可像使用本地接口同樣使用DemoService這個接口裏面的方法:
import org.springframework.context.support.ClassPathXmlApplicationContext; import com.xrq.dubbo.demo.DemoService; public class Consumer { public static void main(String[] args) throws Exception { ClassPathXmlApplicationContext context =
new ClassPathXmlApplicationContext(new String[] {"http://aaa.bbb.ccc:dddd/eee/fff/consumer.xml"}); context.start(); DemoService demoService = (DemoService)context.getBean("demoService"); // 獲取遠程服務代理 String hello = demoService.sayHello("world"); // 執行遠程方法 System.out.println(hello); // 顯示調用結果 } }
看到整個過程當中:
一、沒有任何Dubbo的代碼,使用Dubbo的時候所有都是Spring配置,這就是前面提到的Dubbo對應用沒有任何API侵入
二、不須要考慮底層線程模型、序列化、反序列化、url解析等問題,這些都是Dubbo底層作好的
Dubbo經常使用標籤
首先說一個優先級的問題,在dubbo中好比timeout、retries、loadbalance等參數能夠在多個標籤內同時配置,它們之間的優先級,dubbo手冊上是這麼說的:
OK,下面羅列一些Dubbo經常使用的,也就是說每一個項目的Dubbo的xml文件中基本都會出現的標籤,並以表格形式列舉標籤中常見的可用屬性。
一、<dubbo:service /> 用於服務生產者暴露服務配置
屬 性 | 類 型 | 是否必填 | 缺省值 | 描 述 |
interface | class | 必填 | 無 | 服務接口全路徑 |
ref | object | 必填 | 無 | 服務對象實現引用 |
version | string | 可選 | 0.0.0 | 服務版本,建議使用兩位數字版本如1.0,一般在接口不兼容時版本號才須要升級 |
timeout | int | 可選 | 1000 | 遠程服務調用超時時間(毫秒) |
retries | int | 可選 | 2 | 遠程服務調用重試次數,不包括第一次調用,不須要重試請設爲0 |
connections | int | 可選 | 100 | 每一個生產者的最大鏈接數,短鏈接協議如rmi,表示限制鏈接數;長鏈接協議如dubbo表示創建的長鏈接個數 |
loadbalance | string | 可選 | random | 負載均衡策略,可選值爲:random(隨機)、roundrobin(輪詢)、leastactive(最少活躍調用) |
async | boolean | 可選 | false | 是否缺省異步執行,不可靠的異步,只是忽略返回值,不阻塞執行線程 |
register | boolean | 可選 | true | 該協議的服務是否註冊到註冊中心 |
二、<dubbo:reference /> 用於服務消費者引用服務配置
屬 性 | 類 型 | 是否必填 | 缺省值 | 描 述 |
id | string | 必填 | 無 | 服務引用beanId |
interface | class | 必填 | 無 | 服務接口全路徑 |
version | string | 可選 | 無 | 服務版本,與服務生產者的版本一致 |
timeout | long | 可選 | 使用<dubbo:consumer>的timeout | 服務方法調用超時時間(毫秒) |
retries | int | 可選 | 使用<dubbo:consumer>的retries | 遠程服務調用重試次數,不包括第一次調用,不須要重試請設爲0 |
connections | int | 可選 | 使用<dubbo:consumer>的connections | 每一個生產者的最大鏈接數,短鏈接協議如rmi,表示限制鏈接數;長鏈接協議如dubbo表示創建的長鏈接個數 |
loadbalance | string | 可選 | 使用<dubbo:consumer>的loadbalance | 負載均衡策略,可選值爲:random(隨機)、roundrobin(輪詢)、leastactive(最少活躍調用) |
async | boolean | 可選 | 使用<dubbo:consumer>的async | 是否缺省異步執行,不可靠的異步,只是忽略返回值,不阻塞執行線程 |
check | boolean | 可選 | 使用<dubbo:consumer>的check | 啓動時檢查服務生產者是否存在,true則報錯,false則忽略 |
url | string | 可選 | 無 | 點對點直連服務提供者地址,將繞過註冊中心,好比"dubbo://localhost:20890",這個比較多的使用在測試中 |
protocol | string | 可選 | 無 | 只調用指定協議的服務生產者,其餘協議忽略 |
三、<dubbo:protocol /> 用於服務生產者協議配置(若是須要支持多協議,能夠聲明多個此標籤,並在<dubbo:service />經過protocol屬性指定使用的協議)
屬 性 | 類 型 | 是否必填 | 缺省值 | 描 述 |
id | string | 可選 | dubbo | 協議beanId,<dubbo service />中的protocol引用此ID,若是不填缺省和name屬性值同樣 |
name | sring | 必填 | dubbo | 協議名稱 |
port | int | 可選 | dubbo->20800,rmi->1099,http->80,hessian->80 若是配置爲-1或未配置,則會分配一個沒有被佔用的端口 |
服務端口 |
host | string | 可選 | 自動查找本機ip | 爲空則自動查找本機ip,建議不配置讓Dubbo自動獲取本機ip |
threadpool | string | 可選 | fixed | 線程池類型,可選fixed/cached |
threads | int | 可選 | 100 | 服務線程池大小(固定大小) |
serialization | string | 可選 | dubbo->hession2,rmi->java,http->json | 協議序列化方式,當協議支持多種序列化方式時使用 |
register | boolean | 可選 | true | 該協議的服務是否註冊到註冊中心 |
四、<dubbo:registry /> 用於註冊中心配置(若是有多個不一樣的註冊中心能夠聲明多個標籤而且<dubbo:service />或<dubbo:reference />中使用registry屬性指定)
屬 性 | 類 型 | 是否必填 | 缺省值 | 描 述 |
id | string | 可選 | 無 | 註冊中心引用beanId,可在<dubbo:service />或<dubbo:reference />中引用此ID |
address | string | 必填 | 無 | 註冊中心服務地址,若是地址沒有端口缺省爲9090,同一個集羣內的多個地址用逗號分隔,如:ip:port,ip:port,不一樣的 集羣註冊中心請配置多個<dubbo:registry />標籤 |
protocol | string | 可選 | dubbo | 註冊中心地址協議,支持dubbo、http、local三種協議,分別表示dubbo地址、http地址和本地註冊中心 |
port | int | 可選 | 9090 | 註冊中心缺省端口,當address沒有帶端口時使用此端口做爲缺省值 |
username | string | 可選 | 無 | 登錄註冊中心用戶名,若是註冊中心不須要驗證可不填 |
password | string | 可選 | 無 | 登錄註冊中心密碼,若是註冊中心不須要驗證可不填 |
transport | string | 可選 | netty | 網絡傳輸方式,可選mina、netty |
timeout | int | 可選 | 5000 | 註冊中心請求超時時間(毫秒) |
file | string | 可選 | 無 | 使用文件緩存註冊中心地址列表以及服務提供者列表,應用重啓時將基於此文件恢復,注意兩個註冊中心不能使用同一文件存儲 |
check | boolean | 可選 | true | 註冊中心不存在時,是否報錯 |
register | boolean | 可選 | true | 是否向此註冊中心註冊服務,若是設爲false,將只訂閱,不註冊 |
subscribe | boolean | 可選 | true | 是否向此註冊中心訂閱服務,若是設爲false,將只註冊,不訂閱 |
五、<dubbo:method />用於方法級配置(該標籤爲<dubbo:service/>或<dubbo:reference/>的子標籤,用於控制到方法級)
屬 性 | 類 型 | 是否必填 | 缺省值 | 描 述 |
method | string | 必填 | 無 | 方法名 |
timeout | int | 可選 | 缺省爲<dubbo:reference/>的timeout | 方法調用超時時間(毫秒) |
retires | int | 可選 | 缺省爲<dubbo:reference/>的retries | 遠程服務調用重試次數,不包括第一次調用,不須要重試請設爲0 |
loadbalance | string | 可選 | 缺省爲<dubbo:reference/>的loadbalance | 負載均衡策略,可選值爲:random(隨機)、roundrobin(輪詢)、leastactive(最少活躍調用) |
async | boolean | 可選 | 缺省爲<dubbo:reference/>的async | 是否異步執行,不可靠異步,只是忽略返回值,不阻塞執行線程 |
actives | int | 可選 | 0 | 每服務消費者最大併發調用限制 |
executes | int | 可選 | 0 | 每服務每方法最大使用線程數限制,此屬性只在<dubbo:method/>做爲<dubbo:service/>子標籤時有效 |
另外,還有<dubbo:provider/>和<dubbo:consumer/>分別表示服務提供者(生產者)和服務消費者的缺省值配置,就不列舉了。
Dubbo協議dubbo://
前面有說到一個Dubbo協議,下面來看一下Dubbo協議。
Dubbo協議Dubbo的缺省協議,使用基於mina1.1.7+hessian3.2.1的tbremoting交互。
從上面的適用範圍總結,dubbo適合小數據量大併發的服務調用,以及消費者機器遠大於生產者機器數的狀況,不適合傳輸大數據量的服務好比文件、視頻等,除非請求量很低。
另外,Dubbo手冊還給開發者列出了Dubbo協議使用的約束: