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

来源: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
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...