自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

SpringCloud實(shí)戰(zhàn)小貼士:Zuul的路徑匹配

大數(shù)據(jù)
不論是使用傳統(tǒng)路由的配置方式還是服務(wù)路由的配置方式,我們都需要為每個(gè)路由規(guī)則定義匹配表達(dá)式,也就是上面所說(shuō)的path參數(shù)。在Zuul中,路由匹配的路徑表達(dá)式采用了Ant風(fēng)格定義。

[[206819]]

路徑匹配

不論是使用傳統(tǒng)路由的配置方式還是服務(wù)路由的配置方式,我們都需要為每個(gè)路由規(guī)則定義匹配表達(dá)式,也就是上面所說(shuō)的path參數(shù)。在Zuul中,路由匹配的路徑表達(dá)式采用了Ant風(fēng)格定義。

Ant風(fēng)格的路徑表達(dá)式使用起來(lái)非常簡(jiǎn)單,它一共有下面這三種通配符:

通配符說(shuō)明

我們可以通過(guò)下表的示例來(lái)進(jìn)一步理解這三個(gè)通配符的含義并參考著來(lái)使用:

URL路徑說(shuō)明

/user-service/?它可以匹配/user-service/之后拼接一個(gè)任務(wù)字符的路徑,比如:/user-service/a、/user-service/b、/user-service/c

/user-service/*它可以匹配/user-service/之后拼接任意字符的路徑,比如:/user-service/a、/user-service/aaa、/user-service/bbb。但是它無(wú)法匹配/user-service/a/b

/user-service/**它可以匹配/user-service/*包含的內(nèi)容之外,還可以匹配形如/user-service/a/b的多級(jí)目錄路徑

另外,當(dāng)我們使用通配符的時(shí)候,經(jīng)常會(huì)碰到這樣的問(wèn)題:一個(gè)URL路徑可能會(huì)被多個(gè)不同路由的表達(dá)式匹配上。比如:有這樣的一個(gè)場(chǎng)景,我們?cè)谙到y(tǒng)建設(shè)的一開(kāi)始實(shí)現(xiàn)了user-service服務(wù),并且配置了如下路由規(guī)則:

  1. zuul.routes.user-service.path=/user-service/** 
  2. zuul.routes.user-service.serviceId=user-service 

但是隨著版本的迭代,我們對(duì)user-service服務(wù)做了一些功能拆分,將原屬于user-service服務(wù)的某些功能拆分到了另外一個(gè)全新的服務(wù)user-service-ext中去,而這些拆分的外部調(diào)用URL路徑希望能夠符合規(guī)則/user-service/ext/**,這個(gè)時(shí)候我們需要就在配置文件中增加一個(gè)路由規(guī)則,完整配置如下:

  1. zuul.routes.user-service.path=/user-service/** 
  2. zuul.routes.user-service.serviceId=user-service 
  3.  
  4. zuul.routes.user-service-ext.path=/user-service/ext/** 
  5. zuul.routes.user-service-ext.serviceId=user-service-ext 

這個(gè)時(shí)候,調(diào)用user-service-ext服務(wù)的URL路徑實(shí)際上會(huì)同時(shí)被/user-service/**和/user-service/ext/**兩個(gè)表達(dá)式所匹配。在邏輯上,API網(wǎng)關(guān)服務(wù)需要優(yōu)先選擇/user-service/ext/**路由,然后再匹配/user-service/**路由才能實(shí)現(xiàn)上述需求。但是如果使用上面的配置方式,實(shí)際上是無(wú)法保證這樣的路由優(yōu)先順序的。

從下面的路由匹配算法中,我們可以看到它在使用路由規(guī)則匹配請(qǐng)求路徑的時(shí)候是通過(guò)線性遍歷的方式,在請(qǐng)求路徑獲取到***個(gè)匹配的路由規(guī)則之后就會(huì)返回并結(jié)束匹配過(guò)程。所以當(dāng)存在多個(gè)匹配的路由規(guī)則時(shí),匹配結(jié)果完全取決于路由規(guī)則的保存順序。

  1. @Override 
  2. public Route getMatchingRoute(final String path) { 
  3.     ... 
  4.     ZuulRoute route = null
  5.     if (!matchesIgnoredPatterns(adjustedPath)) { 
  6.         for (Entry<String, ZuulRoute> entry : this.routes.get().entrySet()) { 
  7.             String pattern = entry.getKey(); 
  8.             log.debug("Matching pattern:" + pattern); 
  9.             if (this.pathMatcher.match(pattern, adjustedPath)) { 
  10.                 route = entry.getValue(); 
  11.                 break; 
  12.             } 
  13.         } 
  14.     } 
  15.     log.debug("route matched=" + route); 
  16.     return getRoute(route, adjustedPath); 

下面所示代碼是基礎(chǔ)的路由規(guī)則加載算法,我們可以看到這些路由規(guī)則是通過(guò)LinkedHashMap保存的,也就是說(shuō)路由規(guī)則的保存是有序的,而內(nèi)容的加載是通過(guò)遍歷配置文件中路由規(guī)則依次加入的,所以導(dǎo)致問(wèn)題的根本原因是對(duì)配置文件中內(nèi)容的讀取。

  1. protected Map<String, ZuulRoute> locateRoutes() { 
  2.     LinkedHashMap<String, ZuulRoute> routesMap = new LinkedHashMap<String, ZuulRoute>(); 
  3.     for (ZuulRoute route : this.properties.getRoutes().values()) { 
  4.         routesMap.put(route.getPath(), route); 
  5.     } 
  6.     return routesMap; 

由于properties的配置內(nèi)容無(wú)法保證有序,所以當(dāng)出現(xiàn)這種情況的時(shí)候,為了保證路由的優(yōu)先順序,我們需要使用YAML文件來(lái)配置,以實(shí)現(xiàn)有序的路由規(guī)則,比如使用下面的定義:

  1. zuul: 
  2.   routes: 
  3.     user-service-ext: 
  4.       path: /user-service/ext/** 
  5.       serviceId: user-service-ext 
  6.     user-service: 
  7.       path: /user-service/** 
  8.       serviceId: user-service 

忽略表達(dá)式

通過(guò)path參數(shù)定義的Ant表達(dá)式已經(jīng)能夠完成API網(wǎng)關(guān)上的路由規(guī)則配置功能,但是為了更細(xì)粒度和更為靈活的配置路由規(guī)則,Zuul還提供了一個(gè)忽略表達(dá)式參數(shù):zuul.ignored-patterns。該參數(shù)可以用來(lái)設(shè)置不希望被API網(wǎng)關(guān)進(jìn)行路由的URL表達(dá)式。

比如,以快速入門(mén)中的示例為基礎(chǔ),如果我們不希望/hello接口被路由,那么我們可以這樣設(shè)置:

  1. zuul.ignored-patterns=/**/hello/** 
  2. zuul.routes.api-a.path=/api-a/** 
  3. zuul.routes.api-a.serviceId=hello-service 

然后,可以嘗試通過(guò)網(wǎng)關(guān)來(lái)訪問(wèn)hello-service的/hello接口:http://localhost:5555/api-a/hello。雖然該訪問(wèn)路徑的完全符合path參數(shù)定義的/api-a/**規(guī)則,但是由于該路徑符合zuul.ignored-patterns參數(shù)定義的規(guī)則,所以不會(huì)被正確路由。同時(shí),我們?cè)诳刂婆_(tái)或日志中還能看到?jīng)]有匹配路由的輸出信息:

  1. o.s.c.n.z.f.pre.PreDecorationFilter      : No route found for uri: /api-a/hello 

另外,該參數(shù)在使用時(shí)還需要注意它的范圍并不是對(duì)某個(gè)路由,而是對(duì)所有路由的。所以在設(shè)置的時(shí)候需要全面的考慮URL規(guī)則,防止忽略了不該被忽略的URL路徑。

【本文為51CTO專(zhuān)欄作者“翟永超”的原創(chuàng)稿件,轉(zhuǎn)載請(qǐng)通過(guò)51CTO聯(lián)系作者獲取授權(quán)】

戳這里,看該作者更多好文

責(zé)任編輯:武曉燕 來(lái)源: 51CTO專(zhuān)欄
相關(guān)推薦

2017-04-13 11:06:28

SpringCloud隨機(jī)端口

2017-05-19 15:13:05

過(guò)濾器Spring ClouZuul

2017-05-18 14:14:25

過(guò)濾器Spring ClouZuul

2017-07-31 15:47:50

Zuul統(tǒng)一處理

2017-05-02 23:05:44

HTTPZuulCookie

2017-10-20 14:55:06

Spring ClouZuul加載

2022-05-13 08:23:07

Zuul微服務(wù)Zuul過(guò)濾器

2023-07-17 08:02:44

ZuulIO反應(yīng)式

2020-11-18 11:26:45

SpringCloudZuulJava

2017-08-10 16:14:07

FeignRPC模式

2017-09-26 16:17:39

Ribboneager-load模式

2022-04-09 14:45:02

微服務(wù)常見(jiàn)概念Spring

2021-03-26 06:01:45

日志MongoDB存儲(chǔ)

2021-06-21 08:35:52

AntPathMatcSpringAnt風(fēng)格

2022-05-20 12:14:50

ZuulSpringClou

2013-07-02 11:01:01

2013-10-23 13:01:29

Ubuntu技巧時(shí)光機(jī)

2010-12-14 09:15:50

Visual Stud

2016-12-01 13:50:38

2023-05-29 14:07:00

Zuul網(wǎng)關(guān)系統(tǒng)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)