物流路由線路配載前端演算法邏輯實現方案

来源:https://www.cnblogs.com/Jcloud/archive/2023/04/13/17308853.html
-Advertisement-
Play Games

配載代表著某條線路是否具有發往某個方向(區域、省市縣、分揀等)的能力,也可以說是網點(分揀中心)是否具有承載配載所指方向貨物的能力。一般網路規劃者,在均衡線路間貨量時,會通過調整配載來完成。線路上可允許配載貨物的“產品類型、最終妥投目的地”,通過線路的配載,計算 當前網點 到 目的網點 的 下一個網... ...


作者:京東物流 柳巨集

1.前置知識

1.1 基本概念

1.1.1 配載

  • 配載代表著某條線路是否具有發往某個方向(區域、省市縣、分揀等)的能力,也可以說是網點(分揀中心)是否具有承載配載所指方向貨物的能力。一般網路規劃者,在均衡線路間貨量時,會通過調整配載來完成。
  • 線路上可允許配載貨物的“產品類型、最終妥投目的地”,通過線路的配載,計算 當前網點 到 目的網點 的 下一個網點 ,線路 綁定的配載代表通過當前線路最終可以到達的目的地 。以下圖為例

  • 表示:如果放置在整個路由網路資源中,一個標記T1的貨物要從北京發往福建,可選的路由有①北京站-北京-武漢-福建-福建站;②北京站-北京-廣州-福建-福建站;之所以剔除了北京站-北京-上海-福建-福建站以及北京站-北京-武漢-上海-福建-福建站,正是因為後兩條線路中未包含T1的配載代碼,只標記了T2 ,說明這條線路只有配載航空的貨物,而沒有普通陸運的帶貨能力。
  • 下圖就是用於描述配載的樹形結構

1.1.2 班期與生失效日期

  • 班期:指的是發運頻率,1234567代表著每周七天中,這個班次的“上線時間”,一般來講,維護時缺失某個值,會造成路由中斷的現象。
  • 生失效日期:指的是該配載有效時間範圍

1.1.3 配載合併邏輯

  • 網點四級地址的關係以配載樹的形式展現,勾選節點添加的配載在右側的配載列表中展示

  • 當某個節點的子節點沒有全部勾選時,展示當前勾選的節點到配載列表中

  • 當某個節點的子節點全部勾選時(在符合相關條件時,這裡涉及到的演算法邏輯後面詳述),展示相應的父節點到配載列表中,這個邏輯是遞歸的

1.2 現有實現技術

•目前的線路配載前端基於zTree+FixedHeaderTable+JQuery實現,通過zTree監聽節點被選中和取消選中,計算該操作後是否觸發節點的合併或展開,進而重新渲染配載列表中的數據

2. 現狀問題

2.1 節點合併演算法邏輯有誤

  • 如果一個父節點下的所有子節點都被維護,即使子節點下的班期不同、生失效日期不重疊,系統都會自動合併到父節點。合併的展示效果為:
  • 班期:對於純新增配載顯示1234567;對於父節點下有一個子節點,班期顯示為已存在配載的班期
  • 生失效時間:統一為該切段線路的生失效日期
  • 例如:X線路被切割成兩段,2022-11-022023-01-25以及2022-01-26長久有效兩段,線上路視圖點擊2022-11-02~2023-01-25 這段的配載維護,X下有A1(配載時間為2022-11-02長久有效、班期12),其父節點為A,A下還有子節點A2、A3。今天是11.17日,將A2、A3都勾選上(時間任意,班期為12345),配載會立刻合併為A(生效時間2022-11-022023-01-25,配載1234567)
線路X  
    生效時間        失效時間
    2022-11-02   2023-01-25

A(父節點):包含A1、A2、A3三個子節點,當前只存在A1的配載如下:

    生效時間        失效時間        班期
A1  2022-11-02   2099-12-31      12
           
參考日期為11.17日,此時勾選A2+A3後觸發A的合併,此時配載列表展示A節點

    生效時間        失效時間        班期
A  2022-11-02   2023-01-25      1234567


2.2 配載保存和顯示的值不一致

  • 上面操作觸發合併,提交後庫中保存的配載記錄為:
    生效時間        失效時間        班期
A1  2022-11-02    2022-11-16     12
A   2022-11-17    2023-01-25     1234567


2.3 本質原因

  • 原有根據zTree節點觸發合併的演算法有問題,不考慮當前節點下其他子節點的配載的班期和生失效日期,而是根據是否同一個父節點直接合併。導致合併邏輯錯誤,保存與展示的數據不一致。

3. 預期效果

3.1 配載合併班期邏輯

  • 條件:同一個父節點+各個子節點班期一致
A(父節點):包含A1、A2、A3三個子節點,其中任意節點的班期不一致都無法合併


3.2 配載生失效日期切斷邏輯

  • 新添加節點,生效日期為 參考日期,失效日期為 線路失效日期

  • 參考日期選擇 大於 當前 同級子節點的某天,當觸發合併時
  • 合併後的父節點:生效日期取參考日期,失效時間取同級子節點列表中失效時間最小的
  • 合併後的子節點:

1)如果 原始生效日期小於合併後父節點的生效日期,則切斷 原始生效日期 ~ 父節點的生效日期-1天(相當於保留切斷前的生效日期那一段)

2)如果 原始失效日期大於合併後父節點的失效日期,則切斷 父節點的失效日期+1 ~ 原始失效日期 (相當於保留切斷前的失效日期那一段)

  • 針對同一個節點的配載生失效日期切斷的邏輯,舉例
1. 配載A的原始生失效時間為 20221103 - 20221110
2. 配載A在經過同級子節點合併後,生失效時間為 20221105-20221108
3. 那麼對於配載A來說,在合併後仍然需要保留兩段配載記錄
    3.1 生失效時間為 20221103-20221104
    3.2 生失效時間為 20221109-20221110


3.3 配載合併後保存邏輯

•採用所見即所的方式保存數據,用戶在前端完成切斷操作後,保存到資料庫的記錄與前端展示一致

4. 實現邏輯

4.1 整體邏輯

4.2 定義數據結構及初始化

zTree:配載樹
treeNode:配載樹中的節點
nodeId:節點id
childrenNodes:包含當前節點的所有子節點集合
stowageList:配載列表的Dom結構
originStowageMapTI:原始配載:{key:節點;value:配載數據的dom結構}
newStowageMapTI:新增節點配載:{key:節點id;value:節點}
stowageFrequencyMap:配載節點和班期關係:{key:節點id;value:班期}
stowageTimeMap:配載節點和生失效日期關係:{key:節點id;value:[生效時間,失效時間]}
frequencyTreeMap:班期和節點的關係:{key:班期;value:節點數組}
node.pid:節點的父id
node.id:節點的id


在配載樹上監聽事件,當觸發選中/取消選中時,遞歸的獲取childrenNodes

維護配載與班期、配載與生失效日期的關係

將已有配載列表中的數據維護到stowageFrequencyMap、stowageTimeMap、originStowageMapTI中

4.3 配載合併班期邏輯

1)如果當前節點非禁用 && 勾選 執行 合併邏輯;否則遞歸遍歷節點;最終返回結果集

2)如果當前節點非半選 && 非父節點,向父節點中查找班期,維護節點屬性,加入結果集並返回

3)根據節點的pid查找stowageFrequencyMap中是否存在班期

配載樹中的網點關係主要是四級,舉例說明 C(快遞全國) 下麵某些節點的level 與 pid 的關係

level=1
  pid:C  表示全國

level=2
 pid:C-10 表示華南

level=3
  pid:C-10-16 表示福建

level=4
  pid:C-10-16-1303 表示泉州

演算法邏輯:根據pid截取相應的字元串為key,查找是否存在父節點的班期


1.如果是父節點則遍歷,按照每個子節點去stowageFrequencyMap中獲取班期,分為兩種情況如下;找到班期後維護frequencyTreeMap,將同班期的節點維護到list中保存,即形成 班期-節點list的數據結構

2.向父節點中獲取班期,同上

3.向子節點中遞歸獲取班期,並將結果保存到新的數據結構frequencyChildMap中,然後合併到frequencyTreeMap中

對frequencyTreeMap集合進行判斷,如果數量為1表示 與該節點同級的節點班期相同,觸發了合併,將節點加入結果集返回;否則說明當前節點的同級節點存在不同班期,不能進行節點向上合併,直接遍歷frequencyTreeMap中的value集合,加入結果集返回

對於觸發了合併的結果集 frequencyTreeMap 中的每個節點遍歷 同 當前線路的生失效日期比較,取出同級節點中的最大生效時間最小失效時間作為該同級節點中的最大公共時間範圍設置到每個節點屬性中

4.4 配載生失效日期切斷邏輯

定義變數

1. queryTime:參考日期
2. maxDisableTime:最大失效時間
3. enableTime:線路生效時間
4. disableTime:線路失效時間
5. originEnableTime:記錄原始生效時間
6. originDisableTime:記錄原始失效時間
7. newDisableTime :enableTime - 24 * 60 * 60 * 1000
8. newEnableTime:disableTime + 24 * 60 * 60 * 1000


迴圈遍歷每個結果集中的節點,判斷是否渲染到配載列表中

根據當前節點id判斷是否存在於 stowageList 中,如果存在直接顯示;根據節點id刪除originStowageMapTI集合

更新生失效日期,分別判斷 原始配載中是否存在需要切斷的日期,新添加配載中是否存在需要切斷的日期;然後將當前節點添加到配載列表中

  • 遍歷 originStowageMapTI ,判斷歷史的配載節點是否需要進行日期切斷,分為以下兩種情況;然後根據節點id刪除 originStowageMapTI
  • 如果 originEnableTime < enableTime:添加新的配載記錄,生效時間取 originEnableTime, 失效時間取newDisableTime
  • 如果 originDisableTime > disableTime:添加新的配載記錄,生效時間取 newEnableTime, 失效時間取 originDisableTime

  • 遍歷 newStowageMapTI ,判斷新添加的節點是否需要進行日期切斷;然後根據節點id刪除 newStowageMapTI
  • 如果 originDisableTime > disableTime:添加新的配載記錄,生效時間取 newEnableTime, 失效時間取 originDisableTime

5. 總結

•路由線路配載維護業務核心且頻繁使用功能,為了實現業務述求,將完全沒有關聯的樹形結構Dom列表結合在一起。採用了多種數據模型+數據結構的組合形式,構造兩者之間的關係,結合遍歷、深度優先搜索、字元串查找等演算法進行實現,在春節串點優化專項上線後取得了預期的收益。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 前言 平時在使用v-for的時候,一般會要求傳入key,有沒有像我一樣的小伙伴,為了省心,直接傳索引index,貌似也沒有遇到過什麼問題,直到有一天,我遇到一個這樣的需求 場景 在一個下單界面,我需要去商品列表選商品,然後在下單界面遍歷顯 ...
  • 前端模板 - Anchor UI KIT 前言 今天介紹一款製作精良、開源、免費的 Bootstrap 模板 —— Anchor UI KIT 該模板使用的是Bootstrap v4版本 本文將介紹如何在Django中導入該模板的靜態資源包並使用 介紹 官方文檔 Anchor - a free Bo ...
  • Promise 是非同步編程的一種解決方案,比傳統的回調函數或事件更合理和更靈活。 Promise 方法 Promise的原型方法:then/catch/finally,這三種方法很常用,then用於處理Promise轉為fulfilled狀態時的代碼,catch用於處理Promise轉為reject ...
  • 我們是袋鼠雲數棧 UED 團隊,致力於打造優秀的一站式數據中台產品。我們始終保持工匠精神,探索前端道路,為社區積累並傳播經驗價值。。 本文作者:霜序(掘金) 前言 在我們的業務應用中越來越多的應用到編碼內容,例如在 API 中,給到後端的 SQL 都是通過 Base64 加密的數據等等。 能夠發現我 ...
  • 因為沒看見答案,所以也不知道對不對。 JavaScript 的垃圾回收機制是由 JavaScript 引擎自動管理的,通常情況下我們無法控制垃圾回收機制的執行時間和頻率。 然而,我們可以採取一些優化策略來減少垃圾回收的性能開銷,從而提高代碼執行速度。 減少全局變數:全局變數不易被垃圾回收,因為它們始 ...
  • 本文將介紹如何使用Nuxtjs對vue項目進行ssr和靜態化處理。 Nuxtjs簡單介紹 首先,我們簡單瞭解下Nuxtjs框架,Nuxt.js是一個基於Vue的通用框架,主要用於解決Vue項目的服務端渲染(SSR)。 它本質上是一個Vue框架,增加一層node服務,通過對客戶端/服務端的抽象封裝,使 ...
  • 今日,群友提問,如何實現這麼一個 Loading 效果: 這個確實有點意思,但是這是 CSS 能夠完成的? 沒錯,這個效果中的核心氣泡效果,其實藉助 CSS 中的濾鏡,能夠比較輕鬆的實現,就是所需的元素可能多點。參考我們之前的: 使用純 CSS 實現超酷炫的粘性氣泡效果 巧用 CSS 實現酷炫的充電 ...
  • Web 前端自動化測試是一種可以提高測試效率、減少測試成本和提高測試質量的方法,適用於各種類型的 Web 應用程式。本文談談前端自動化測試從入門到精通再到專家級的方案與思維! ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...