2023-08-11 10:31:15來源:Spring全家桶實戰(zhàn)案例源碼
環(huán)境:springcloud Hoxton.SR11
(資料圖片)
本節(jié)主要了解系統(tǒng)中的謂詞與配置的路由信息是如何進行初始化關聯(lián)生成路由對象的。每個謂詞工廠中的Config對象又是如何被解析配置的。
所有的謂詞工廠中的Config中屬性值是如何被配置的。
在SpringCloud Gateway中的所有謂詞工廠如下:
圖片
命名規(guī)則:XxxRoutePredicateFactory。所有的這些謂詞工廠都是如下的繼承關系
public class MethodRoutePredicateFactory extends AbstractRoutePredicateFactory// public class PathRoutePredicateFactory extends AbstractRoutePredicateFactory// ...
所有的謂詞工廠繼承的AbstractRoutePredicateFactory中的泛型都是內(nèi)部類的Config。這個是如何被配置上值的呢?
1.1 gateway自動配置在下面這個類中配置了所有的Predicate和Filter。
public class GatewayAutoConfiguration { @Bean @ConditionalOnEnabledPredicate public PathRoutePredicateFactory pathRoutePredicateFactory() { return new PathRoutePredicateFactory(); } @Bean @ConditionalOnEnabledPredicate public QueryRoutePredicateFactory queryRoutePredicateFactory() { return new QueryRoutePredicateFactory(); } @Bean public RouteLocator routeDefinitionRouteLocator(GatewayProperties properties, List gatewayFilters, List predicates, RouteDefinitionLocator routeDefinitionLocator, ConfigurationService configurationService) { return new RouteDefinitionRouteLocator(routeDefinitionLocator, predicates, gatewayFilters, properties, configurationService); } @Bean @Primary @ConditionalOnMissingBean(name = "cachedCompositeRouteLocator") public RouteLocator cachedCompositeRouteLocator(List routeLocators) { return new CachingRouteLocator(new CompositeRouteLocator(Flux.fromIterable(routeLocators))); }}
這里會層層委托最終查找查找路由定位會交給RouteDefinitionRouteLocator。CachingRouteLocator起到緩存的作用,將配置的所有路由信息保存。
注意:這里的路由信息是在容器啟動后就會被初始化的。
public class CachingRouteLocator { private final RouteLocator delegate; private final Flux routes; private final Map cache = new ConcurrentHashMap<>(); private ApplicationEventPublisher applicationEventPublisher; public CachingRouteLocator(RouteLocator delegate) { this.delegate = delegate; routes = CacheFlux.lookup(cache, CACHE_KEY, Route.class) .onCacheMissResume(this::fetch); } private Flux fetch() { return this.delegate.getRoutes().sort(AnnotationAwareOrderComparator.INSTANCE); }}
實例化CachingRouteLocator就開始查找所有配置的Route信息。最終的會委托給RouteDefinitionRouteLocator
RouteDefinitionRouteLocator構造函數(shù)中的initFactories方法用來映射路由工廠的XxxRoutePredicateFactory。
private void initFactories(List predicates) { predicates.forEach(factory -> { String key = factory.name(); if (this.predicates.containsKey(key)) { this.logger.warn("A RoutePredicateFactory named " + key + " already exists, class: " + this.predicates.get(key) + ". It will be overwritten."); } this.predicates.put(key, factory); });}
方法中解析每一個謂詞工廠對應的名稱然后緩存到predicates 集合中。
factory.name()方法解析謂詞名稱。
default String name() { return NameUtils.normalizeRoutePredicateName(getClass());}
CachingRouteLocator是個緩存路由定位器,是個首選的RouteLocator(@Primary),這里將RouteDefinitionRouteLocator進行了合并。
1.2 生成路由對象Route及Config配置getRoutes---》convertToRoute---》combinePredicates---》lookup。
根據(jù)上面的自動配置也知道了在服務啟動時就進行初始化所有路由信息了。
獲取路由信息
public Flux getRoutes() { Flux routes = this.routeDefinitionLocator.getRouteDefinitions() .map(this::convertToRoute); routes = routes.onErrorContinue((error, obj) -> { return routes.map(route -> { return route; });}
合并謂詞(鏈式調(diào)用)
private AsyncPredicate combinePredicates( RouteDefinition routeDefinition) { // other code for (PredicateDefinition andPredicate : predicates.subList(1, predicates.size())) { AsyncPredicate found = lookup(routeDefinition, andPredicate); predicate = predicate.and(found); } return predicate;}
進入lookup中
private AsyncPredicate lookup(RouteDefinition route, PredicateDefinition predicate) { RoutePredicateFactory
lookup方法中查找,也就是在這里將對應的謂詞Config與RouteDefinition(Predicate)中定義的相對應的屬性關聯(lián)。
進入factory.applyAsync方法
@FunctionalInterfacepublic interface RoutePredicateFactory extends ShortcutConfigurable, Configurable { default AsyncPredicate applyAsync(C config) { return toAsyncPredicate(apply(config)); // 查看下面的6.2-1圖當前apply所有的實現(xiàn)就是系統(tǒng)內(nèi)部定義的XxxRoutePredicateFactory }}// apply(config),如這里配置了Path謂詞,那么就會進入PathRoutePredicateFactory中的apply方法public Predicate apply(Config config) { // other code return new GatewayPredicate() { public boolean test() { // todo } }}// 最后返回一個異步的謂詞public static AsyncPredicate toAsyncPredicate(Predicate super ServerWebExchange> predicate) { Assert.notNull(predicate, "predicate must not be null"); // 這里from就是返回一個DefaultAsyncPredicate默認的異步謂詞 return AsyncPredicate.from(predicate);}static AsyncPredicate from( Predicate super ServerWebExchange> predicate) { return new DefaultAsyncPredicate<>(GatewayPredicate.wrapIfNeeded(predicate));}
圖6.2-1
圖片
最后在combinePredicates方法中將當前路由中配置的所有謂詞進行了and操作返回。最終回到convertToRoute方法中將當前路由中配置的謂詞,過濾器進行了整合包裝返回Route(一個路由對象)
public class Route implements Ordered { private final String id; private final URI uri; private final int order; private final AsyncPredicate predicate; private final List gatewayFilters; private final Map metadata;}
這些Route對象會被保存在上面說的CachingRouteLocator.routes中。
6.3 定位路由根據(jù)上面的配置RouteLocator 該類用來定位路由(查找具體的使用哪個路由);當一個請求過來會查找是哪個路由。
RouteLocator中定義了一個方法
public interface RouteLocator { Flux getRoutes();}
查看這個getRoutes方法是誰調(diào)用的
圖片
看到這個RoutePredicateHandlerMapping是不是想起了Spring MVC中的HandlerMapping(我們所有的Controller都會被 RequestMappingHanlderMapping 匹配)。通過名稱也就知道了該HandlerMapping用來匹配我們的路由謂詞的誰來處理路由。
接下來回到前面說的RequestMappingHanlderMapping 對象,當我們請求一個路由地址時會執(zhí)行該類中的lookup方法查找路由
protected Mono lookupRoute(ServerWebExchange exchange) { // 這里的this.routeLocator就是 CachingRouteLocator對象 return this.routeLocator.getRoutes() .concatMap(route -> Mono.just(route).filterWhen(r -> { exchange.getAttributes().put(GATEWAY_PREDICATE_ROUTE_ATTR, r.getId()); // 過濾查找符合的路由 return r.getPredicate().apply(exchange); }).doOnError(e -> logger.error( "Error applying predicate for route: " + route.getId(), e)).onErrorResume(e -> Mono.empty())) .next() .map(route -> { if (logger.isDebugEnabled()) { logger.debug("Route matched: " + route.getId()); } validateRoute(route, exchange); return route; });}
進入r.getPredicate().apply(exchange)
public interface AsyncPredicate extends Function> { static AsyncPredicate from(Predicate super ServerWebExchange> predicate) { return new DefaultAsyncPredicate<>(GatewayPredicate.wrapIfNeeded(predicate)); } class DefaultAsyncPredicate implements AsyncPredicate { private final Predicate delegate; public DefaultAsyncPredicate(Predicate delegate) { this.delegate = delegate; } @Override public Publisher apply(T t) { return Mono.just(delegate.test(t)); } @Override public String toString() { return this.delegate.toString(); } }}
這里會調(diào)用Predicate.test方法(XxxRoutePredicateFactory中的apply方法返回的GatewayPredicate)。
調(diào)用GatewayPredicate.test返回判斷當前請求的路由是否匹配。
整體的一個流程:
1、系統(tǒng)先初始化所有的Predicate(謂詞)和Filter(過濾器)
2、根據(jù)配置的路由信息(過濾器,謂詞)包裝返回Route對象
3、根據(jù)請求路由路徑查找匹配的路由
關鍵詞:
環(huán)境:springcloudHoxton SR11本節(jié)主要了解系統(tǒng)中的謂詞與配置的路由信
近日,中國東航宣布“空中Wi-Fi”服務突破3000米以下限制,從起飛到落
近段時間以來,伴隨資金持續(xù)流入,多只港股相關ETF份額持續(xù)增長,其中
sql_helper工具簡介索引在數(shù)據(jù)庫中非常重要,它可以加快查詢速度并提高
08月10日,德明利被深股通減持5 87萬股,最新持股量為37 11萬股,占公
香城體育中心公共區(qū)域的籃球場,香城體育中心小朋友在羽毛球館訓練,猛
將首都從爪哇島上的雅加達遷往加里曼島上的努桑塔拉,印尼總統(tǒng)佐科最近
文章稱,面對數(shù)字經(jīng)濟帶來的機遇和挑戰(zhàn),資本市場各方勇立潮頭,積極開
1、1、您的設置是正確無誤的。360版的EASTNOD32都是這么個設置的。網(wǎng)絡
0471房產(chǎn)來為大家解答以上的問題。謝瑤環(huán)故事簡介,歷史上的謝瑤環(huán)簡介
美國百年歷史小鎮(zhèn)被毀,夏威夷山火已致36死:格隆匯8月10日|8月9日,
中新網(wǎng)日喀則8月10日電題:西藏日喀則扎西吉彩:讓非遺“特技”在新時
8月9日,國家游泳中心里充滿了小朋友們的歡笑聲,“關愛成長點亮希望”
8月10日,據(jù)彭博社今日報道稱,證監(jiān)會計劃于本周五,召集部分房企和金
快科技8月10日消息,今天,vivo官方宣布,新款平板vivoPadAir將在8月14