From b894ef7ebdf70daa1b929b8afbf0840fcf5bd3e0 Mon Sep 17 00:00:00 2001 From: skirtle <65301168+skirtles-code@users.noreply.github.com> Date: Sat, 10 Feb 2024 14:47:24 +0000 Subject: [PATCH] perf: use a binary search for insertMatcher --- packages/router/src/matcher/index.ts | 65 ++++++++++++++++++++-------- 1 file changed, 47 insertions(+), 18 deletions(-) diff --git a/packages/router/src/matcher/index.ts b/packages/router/src/matcher/index.ts index 39a3c24b1..9ad6f2bc1 100644 --- a/packages/router/src/matcher/index.ts +++ b/packages/router/src/matcher/index.ts @@ -223,17 +223,8 @@ export function createRouterMatcher( } function insertMatcher(matcher: RouteRecordMatcher) { - let i = 0 - while ( - i < matchers.length && - comparePathParserScore(matcher, matchers[i]) >= 0 && - // Adding children with empty path should still appear before the parent - // https://github.com/vuejs/router/issues/1124 - (matcher.record.path !== matchers[i].record.path || - !isRecordChildOf(matcher, matchers[i])) - ) - i++ - matchers.splice(i, 0, matcher) + const index = findInsertionIndex(matcher, matchers) + matchers.splice(index, 0, matcher) // only add the original record to the name map if (matcher.record.name && !isAliasRecord(matcher)) matcherMap.set(matcher.record.name, matcher) @@ -520,13 +511,51 @@ function checkMissingParamsInAbsolutePath( } } -function isRecordChildOf( - record: RouteRecordMatcher, - parent: RouteRecordMatcher -): boolean { - return parent.children.some( - child => child === record || isRecordChildOf(record, child) - ) +/** + * Performs a binary search to find the correct insertion index for a new matcher. + * + * Matchers are primarily sorted by their score. If scores are tied then the matcher's depth is used instead. + * The depth check ensures that a child with an empty path comes before its parent. + * + * @param matcher - new matcher to be inserted + * @param matchers - existing matchers + */ +function findInsertionIndex( + matcher: RouteRecordMatcher, + matchers: RouteRecordMatcher[] +) { + const depth = getDepth(matcher) + + let lower = 0 + let upper = matchers.length + + while (lower !== upper) { + const mid = (lower + upper) >> 1 + const sortOrder = + comparePathParserScore(matcher, matchers[mid]) || + getDepth(matchers[mid]) - depth + + if (sortOrder === 0) { + return mid + } else if (sortOrder < 0) { + upper = mid + } else { + lower = mid + 1 + } + } + + return upper +} + +function getDepth(record: RouteRecordMatcher) { + let depth = 0 + + while (record.parent) { + ++depth + record = record.parent + } + + return depth } export type { PathParserOptions, _PathParserOptions }