主頁 > 後端開發 > 最佳實踐:路徑路由匹配規則的設計與實作

最佳實踐:路徑路由匹配規則的設計與實作

2023-05-11 07:25:44 後端開發

最佳實踐:路徑路由匹配規則的設計與實作

作者:哲思
時間:2023.5.9
郵箱:[email protected]
GitHub:zhe-si (哲思) (github.com)

前言

時間一晃研究生都過去大半年了,學了些東西,也做了些專案,借著博客總結一下,這次先聊一個簡單的話題開個頭,

開發中,常用形似 “a/b/c” 的描述方式來描述路徑、定位資源,有著層次化和可讀性高的特點,最經典的例子就是 URL(統一資源定位符),第二節會進行簡要介紹,

將資源都路徑化后,可以通過每一段路徑精確的匹配來唯一的確定一個資源,但有時候,需要對具有相關特征的一組資源進行統一的描述或操作,比如,將所有獲得用戶資訊的請求都路由到一個指定的處理程式上,請求的 URL 中包含不同用戶 id 路徑分段指向不同用戶資訊資源,再比如,界面中導航欄包含圖片組(包含圖1、2、3)和文本組(包含文本1、2、3),在訪問圖片組下不同圖片時打開圖片展示器而在訪問文本組的文本時打開文本展示器,

基于上述場景的需求,需要一種簡單而通用的路徑路由匹配規則,最強大的方式是直接使用正則運算式來描述一組路徑,但在描述一些復雜的路徑場景時,正則運算式使用起來非常繁瑣和困難,比如,匹配這樣一組路徑 "x1/a/x2/a",x1 表示任意長的最短匹配路徑,x2表示任意長的最長匹配路徑,大家可以嘗試用正則運算式實作,并和本文設計的匹配規則的描述進行對比,

本文設計并實作了一種專用于路徑路由匹配的規則,以一種簡單而通用的方式描述一組路徑的特征,來簡化這種場景路由描述難度,讓小白可以快速學習并上手,

什么是URL?什么是路徑?

首先,需要明確一下什么是資源?什么是路徑?

上面提到的 URL(統一資源定位符)是 URI(統一資源識別符號)的一種分類,

URI 的本質語意是標識一個資源,資源可以是一張圖片、一個檔案、一個服務、一個用戶等具體或抽象的物體,官方(RFC2396)將其格式標準化為如下格式(就是 URL 的格式)

URI格式

該格式的大致含義是某人(user:pass)用某種方式(protocol)訪問某個主機埠(hostname:port)某個路徑(pathName)的資源,同時可用 search 對該資源做篩選、排序等操作、用 hash 訪問資源的片段(子資源),

而標識一個資源,可以通過描述位置或名字的方式,所以 URI 包括 URL 和 URN(統一資源名稱),

  • 描述位置:用資源所處的地址來描述該資源,該描述指定了在特定地址的資源而不特指某一個具體的資源,也就是說實際指向的資源可能會隨時間發生變化,資源的位置描述也會隨資源本身位置的變化而變化,如 URL,
  • 描述名字:用一個全域唯一的識別符號持久的標記一個特定的資源,不會隨著時間或位置變化而改變指向的資源,如 URN(例:urn:oasis:names:specification:docbook:dtd:xml:4.1.2),常用于 Map、Redis 中 KEY 的定義等場景,

但不管是位置描述還是名字描述、不管具體的格式是什么,都可以把它們抽象為一種“路徑”,只是路徑的描述的含義不同、分隔符不同,

比如,URL 中,最核心的部分就是hostname:port/path這一部分,如下圖藍色區域,

url與路徑

藍色區域已經完整描述了資源的位置,protocol 是補充描述了訪問資源的方式,username:password 是附帶的認證資訊,search 和 hash 則描述的對某一個資源的進一步處理,而hostname:port/path就是一個路徑,每個路徑分段描述的是某個層級的位置節點,

比如,URN 中,路徑的每個路徑分段則描述不同命名空間及命名空間下的名字,

路徑路由匹配規則的設計

了解了什么是路徑,接下來給出路徑路由匹配規則的定義描述:

  • R 模式:正則模式,格式:R:正則運算式

    該模式下,完全采用正則運算式格式進行匹配,

  • 標準模式:標準路徑路由描述運算式,格式形如:/**/xxx/*/xxx,由多個路徑分段的分段串列組成,要求路徑分段串列全匹配

    具體語法:

    • 分隔方法

      默認使用 '/' 分隔符分隔多個路徑分段

      • 支持自定義路由分隔方法路徑分隔方法
    • 路徑分段

      • 每個具體的路徑分段默認采用完全字串匹配
      • r 模式:路徑分段正則模式,該路徑分段采用正則運算式進行匹配,格式:r:正則運算式
    • 通配符 '?'

      匹配任意一段或 0 段路徑分段,在滿足后續部分匹配的情況下優先不匹配

    • 通配符 '*'

      匹配任意一端路徑,不可匹配慷訓不匹配

    • 通配符 '**'

      匹配任意多段路徑分段,不保證盡可能滿足的最短匹配原則,即在滿足緊接的后續非通配符部分匹配的情況下盡可能少的匹配

    • 通配符 '***'

      匹配任意多段路徑分段,保證盡可能滿足的最長匹配原則,即在滿足后續匹配的情況下盡可能多的匹配

舉一些例子:

路由 匹配路徑 不匹配路徑
a/?/c a/b/c、a//c、a/c a/c/d
a/*/c a/b/c a/c
a/b/* a/b/c a/b
**/b/c a/b/c、b/c、a/a/b/b/c b/c/b/c
a/***/c/* a/c/c、a/c/b/c/d a/b/c
a/**/c/* a/c/c a/c/b/c/d

一組路徑 "x1/a/x2/a",x1 表示任意長的最短匹配路徑,x2表示任意長的最長匹配路徑,使用標準路徑路由描述運算式描述就是**/a/***/b

其中,最常用的通配符是 "**",通過不保證盡可能匹配的方式最短匹配,確保匹配的是我們直觀預期的路徑,比如如下目錄結構,

- common
	- A.java
	- B.java
	- a.conf
	- impl
		- common
			- Utils.java
		- AImpl.java
		- BImpl.java

我們希望匹配介面 A 和 B 的 java 檔案,而不匹配到 impl 里的實作類,可以采用如下匹配方式:**/common/r:.*\.java

路徑路由匹配規則的實作

本文采用 kotlin 進行實作,重點位置已經進行注釋說明,源代碼可見倉庫,

/**
 * **路由匹配**
 * - 若 "R:" 開頭,則為正則模式,采用正則運算式直接匹配
 * - 其他情況,采用標準路由模式[matchStdRoutePattern]進行匹配
 *
 * @author lq
 * @version 1.0
 */
fun matchRoutePattern(routePattern: String, path: String): Boolean {
    return if (routePattern.startsWith("R:")) {
        Regex(routePattern.substring(2)).matches(path)
    } else {
        matchStdRoutePattern(routePattern, path)
    }
}

/**
 * **路由模式**:路由的特定描述運算式,形如 / ** /xxx/ * /xxx/
 *
 * 語法:
 * - 以 '/'([PATH_DELIMITER]) 分隔的路徑運算式,要求全匹配路徑,首尾的分隔符可以省略
 * - 每一段路徑描述默認采用字串完全匹配方式,也可通過 "r:" 開頭標記該段采用正則運算式匹配
 * - 使用通配符 "?" 可以匹配任意一段或 0 段路徑,優先不匹配
 * - 使用通配符 "*" 可以匹配任意一段路徑
 * - 使用通配符 "**" 可以匹配任意多段路徑,最短匹配原則
 * - 使用通配符 "***" 可以匹配任意多段路徑,最長匹配原則
 */
fun matchStdRoutePattern(routePattern: String, path: String): Boolean {
    val routeSplit = splitRoute(routePattern)
    val pathSplit = splitPath(path)
    return matchRoutePatternSplit(routeSplit, 0, pathSplit, 0)
}

/**
 * 路由分隔方法
 */
val splitRoute: (String) -> List<String> = ::splitPathSimple
/**
 * 路徑分隔方法
 */
val splitPath: (String) -> List<String> = ::splitPathSimple

/**
 * 簡單決議路徑為路徑分段串列
 */
private fun splitPathSimple(routePattern: String): List<String> {
    val pathDelimiter = '/'
    return routePattern.trim(pathDelimiter).split(pathDelimiter).filter { p -> p.isNotEmpty() }
}

private fun matchRoutePatternSplit(routeSplit: List<String>, ri: Int, pathSplit: List<String>, pi: Int): Boolean {
    if (ri >= routeSplit.size) {
        return pi >= pathSplit.size
    }
    if (pi >= pathSplit.size) {
        for (i in ri until routeSplit.size) {
            if (routeSplit[i] !in listOf("?", "**", "***")) return false
        }
        return true
    }
    when (routeSplit[ri].trim()) {
        "?" -> {
            if (matchRoutePatternSplit(routeSplit, ri + 1, pathSplit, pi)) return true
            return matchRoutePatternSplit(routeSplit, ri + 1, pathSplit, pi + 1)
        }
        "*" -> {
            return matchRoutePatternSplit(routeSplit, ri + 1, pathSplit, pi + 1)
        }
        "**" -> {
            for (i in 0 until pathSplit.size - pi) {
                val isShortMatch = matchRoutePatternShort(routeSplit, ri + 1, pathSplit, pi + i, false)
                if (isShortMatch.first) return true
                if (isShortMatch.second) return false
            }
            return matchRoutePatternSplit(routeSplit, ri + 1, pathSplit, pi + pathSplit.size - pi)
        }
        "***" -> {
            for (i in pathSplit.size - pi downTo 1) {
                if (matchRoutePatternSplit(routeSplit, pi + 1, pathSplit, pi + i)) return true
            }
            return matchRoutePatternSplit(routeSplit, ri + 1, pathSplit, pi)
        }
        else -> {
            if (!checkRouteSegPattern(routeSplit[ri], pathSplit[pi])) return false
            return matchRoutePatternSplit(routeSplit, ri + 1, pathSplit, pi + 1)
        }
    }
}

/**
 * 最短原則匹配,回傳 (是否匹配, 是否已經最短匹配)
 */
private fun matchRoutePatternShort(routeSplit: List<String>, ri: Int, pathSplit: List<String>, pi: Int, isShortMatch: Boolean): Pair<Boolean, Boolean> {
    if (ri >= routeSplit.size) {
        return (pi >= pathSplit.size) to isShortMatch
    }
    if (pi >= pathSplit.size) {
        for (i in ri until routeSplit.size) {
            if (routeSplit[i] !in listOf("?", "**", "***")) return false to isShortMatch
        }
        return true to isShortMatch
    }
    when (routeSplit[ri].trim()) {
        "?" -> {
            val isMatch = matchRoutePatternShort(routeSplit, ri + 1, pathSplit, pi, isShortMatch)
            if (isMatch.first) return isMatch
            return matchRoutePatternShort(routeSplit, ri + 1, pathSplit, pi + 1, isShortMatch)
        }
        "*" -> {
            return matchRoutePatternShort(routeSplit, ri + 1, pathSplit, pi + 1, isShortMatch)
        }
        "**" -> {
            return matchRoutePatternSplit(routeSplit, ri, pathSplit, pi) to isShortMatch
        }
        "***" -> {
            return matchRoutePatternSplit(routeSplit, ri, pathSplit, pi) to isShortMatch
        }
        else -> {
            return if (checkRouteSegPattern(routeSplit[ri], pathSplit[pi])) {
                matchRoutePatternShort(routeSplit, ri + 1, pathSplit, pi + 1, true)
            } else {
                false to isShortMatch
            }
        }
    }
}

/**
 * 路徑分段匹配檢查
 */
private fun checkRouteSegPattern(routeSeg: String, pathSeg: String): Boolean {
    if (routeSeg.startsWith("r:")) {
        if (Regex(routeSeg.substring(2)).matches(pathSeg)) {
            return true
        }
    } else if (routeSeg == pathSeg) return true
    return false
}

后記

本次分享了在專案中的一個細節設計,后續會繼續分享在作業、學習和生活中的點點滴滴,也歡迎大家在評論區共同討論或與我郵件溝通,

本文來自博客園,作者:_哲思,轉載請注明原文鏈接:https://www.cnblogs.com/zhe-si/p/17388551.html

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/552099.html

標籤:其他

上一篇:文盤Rust —— rust連接oss | 京東云技術團隊

下一篇:返回列表

標籤雲
其他(158769) Python(38125) JavaScript(25413) Java(18025) C(15225) 區塊鏈(8263) C#(7972) AI(7469) 爪哇(7425) MySQL(7175) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5871) 数组(5741) R(5409) Linux(5338) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4570) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2432) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) .NET技术(1972) 功能(1967) Web開發(1951) HtmlCss(1935) python-3.x(1918) 弹簧靴(1913) C++(1913) xml(1889) PostgreSQL(1875) .NETCore(1860) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • 最佳實踐:路徑路由匹配規則的設計與實作

    本文設計并實作了一種專用于路徑路由匹配的規則,以一種簡單而通用的方式描述一組路徑的特征,來簡化這種場景路由描述難度,讓小白可以快速學習并上手。 ......

    uj5u.com 2023-05-11 07:25:44 more
  • 文盤Rust —— rust連接oss | 京東云技術團隊

    物件存盤是云的基礎組件之一,各大云廠商都有相關產品。這里跟大家介紹一下rust與物件存盤交到的基本套路和其中的一些技巧。 ......

    uj5u.com 2023-05-10 10:44:00 more
  • Java的抽象類 & 介面

    抽象類:在子類繼承父類時,父類的一些方法實作是不明確的(父類對子類的實作一無所知)。這時需要使父類是抽象類,在子類中提供方法的實作。

    介面(interface)技術主要用來描述類具有什么功能,而并不給出每個功能的具體實作。 ......

    uj5u.com 2023-05-10 10:38:48 more
  • 【11個適合畢設的Python可視化大屏】用pyecharts開發拖拽式可視

    你好,我是@馬哥python說,一枚10年程式猿。 一、效果演示 以下是我近期用Python開發的原創可視化資料分析大屏,非常適合畢設用,下面逐一展示:(以下是截圖,實際上有動態互動效果哦) 以下大屏均為@馬哥python說的個人原創,請勿轉載。 1.1 影視劇分析大屏 1.2 豆瓣電影分析大屏A ......

    uj5u.com 2023-05-10 10:27:16 more
  • 【11個適合畢設的Python可視化大屏】用pyecharts開發拖拽式可視

    你好,我是@馬哥python說,一枚10年程式猿。 一、效果演示 以下是我近期用Python開發的原創可視化資料分析大屏,非常適合畢設用,下面逐一展示:(以下是截圖,實際上有動態互動效果哦) 以下大屏均為@馬哥python說的個人原創,請勿轉載。 1.1 影視劇分析大屏 1.2 豆瓣電影分析大屏A ......

    uj5u.com 2023-05-10 10:25:24 more
  • Java的抽象類 & 介面

    抽象類:在子類繼承父類時,父類的一些方法實作是不明確的(父類對子類的實作一無所知)。這時需要使父類是抽象類,在子類中提供方法的實作。

    介面(interface)技術主要用來描述類具有什么功能,而并不給出每個功能的具體實作。 ......

    uj5u.com 2023-05-10 10:24:35 more
  • C語言快速入門教程1快速入門 2指令 3條件選擇

    快速入門 什么是C語言? C是一種編程語言,1972年由Dennis Ritchie在美國AT & T的貝爾實驗室開發。C語言變得很流行,因為它很簡單,很容易使用。今天經常聽到的一個觀點是--"C語言已經被C++、Python和Java等語言所取代,所以今天何必再去學習C語言"。我很不贊同這種觀點。 ......

    uj5u.com 2023-05-10 10:23:23 more
  • 高效c語言1快速入門

    本章將開發你的第一個C語言程式:傳統的 "Hello, world!"程式。然后討論一些編輯器和編譯器的選項,并闡述移植性問題。 Hello, world! #include <stdio.h> #include <stdlib.h> int main(void) { puts("Hello, wo ......

    uj5u.com 2023-05-10 10:22:59 more
  • 怎樣成為優秀的后端工程師

    本文翻譯自國外論壇 medium,原文地址:https://medium.com/@pradeesh-kumar/how-to-become-a-good-backend-engineer-9da75202a104 讓我們一起看看國外開發者認為優秀后端工程師需要掌握哪些技能。 誰是后端工程師? 本質 ......

    uj5u.com 2023-05-10 07:38:38 more
  • Django筆記三十九之settings配置介紹

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記三十九之settings配置介紹 這一篇筆記介紹 Django 里 settings.py 里一些常用的配置項,這些配置有一些是在之前的筆記中有過介紹的,比如 logging 的日志配置,session 的會話配置等,這里就只做一下簡單的 ......

    uj5u.com 2023-05-10 07:37:28 more