From 731ef56e1069dd2116f7aa2627dca23631e5707f Mon Sep 17 00:00:00 2001 From: Revone Date: Tue, 3 Dec 2024 15:15:48 +1300 Subject: [PATCH] docs: Add an example code for a sliding window using Deque. --- src/data-structures/queue/deque.ts | 88 +++++++++++++++++++ test/unit/data-structures/queue/deque.test.ts | 37 ++++++++ 2 files changed, 125 insertions(+) diff --git a/src/data-structures/queue/deque.ts b/src/data-structures/queue/deque.ts index 4cbb437..b4ac0aa 100644 --- a/src/data-structures/queue/deque.ts +++ b/src/data-structures/queue/deque.ts @@ -15,6 +15,94 @@ import { calcMinUnitsRequired, rangeCheck } from '../../utils'; * 3. Continuous Memory Allocation: Since it is based on an array, all elements are stored contiguously in memory, which can bring cache friendliness and efficient memory access. * 4. Efficiency: Adding and removing elements at both ends of a deque is usually very fast. However, when the dynamic array needs to expand, it may involve copying the entire array to a larger one, and this operation has a time complexity of O(n). * 5. Performance jitter: Deque may experience performance jitter, but DoublyLinkedList will not + * @example + * // prize roulette + * class PrizeRoulette { + * private deque: Deque; + * + * constructor(prizes: string[]) { + * // Initialize the deque with prizes + * this.deque = new Deque(prizes); + * } + * + * // Rotate clockwise to the right (forward) + * rotateClockwise(steps: number): void { + * const n = this.deque.length; + * if (n === 0) return; + * + * for (let i = 0; i < steps; i++) { + * const last = this.deque.pop(); // Remove the last element + * this.deque.unshift(last!); // Add it to the front + * } + * } + * + * // Rotate counterclockwise to the left (backward) + * rotateCounterClockwise(steps: number): void { + * const n = this.deque.length; + * if (n === 0) return; + * + * for (let i = 0; i < steps; i++) { + * const first = this.deque.shift(); // Remove the first element + * this.deque.push(first!); // Add it to the back + * } + * } + * + * // Display the current prize at the head + * display() { + * return this.deque.first; + * } + * } + * + * // Example usage + * const prizes = ['Car', 'Bike', 'Laptop', 'Phone', 'Watch', 'Headphones']; // Initialize the prize list + * const roulette = new PrizeRoulette(prizes); + * + * // Display the initial state + * console.log(roulette.display()); // 'Car' // Car + * + * // Rotate clockwise by 3 steps + * roulette.rotateClockwise(3); + * console.log(roulette.display()); // 'Phone' // Phone + * + * // Rotate counterclockwise by 2 steps + * roulette.rotateCounterClockwise(2); + * console.log(roulette.display()); // 'Headphones' + * @example + * // sliding window + * // Maximum function of sliding window + * function maxSlidingWindow(nums: number[], k: number): number[] { + * const n = nums.length; + * if (n * k === 0) return []; + * + * const deq = new Deque(); + * const result: number[] = []; + * + * for (let i = 0; i < n; i++) { + * // Delete indexes in the queue that are not within the window range + * if (deq.length > 0 && deq.first! === i - k) { + * deq.shift(); + * } + * + * // Remove all indices less than the current value from the tail of the queue + * while (deq.length > 0 && nums[deq.last!] < nums[i]) { + * deq.pop(); + * } + * + * // Add the current index to the end of the queue + * deq.push(i); + * + * // Add the maximum value of the window to the results + * if (i >= k - 1) { + * result.push(nums[deq.first!]); + * } + * } + * + * return result; + * } + * + * const nums = [1, 3, -1, -3, 5, 3, 6, 7]; + * const k = 3; + * console.log(maxSlidingWindow(nums, k)); // [3, 3, 5, 5, 6, 7] */ export class Deque extends IterableElementBase> { /** diff --git a/test/unit/data-structures/queue/deque.test.ts b/test/unit/data-structures/queue/deque.test.ts index 1628d6d..44ec317 100644 --- a/test/unit/data-structures/queue/deque.test.ts +++ b/test/unit/data-structures/queue/deque.test.ts @@ -776,4 +776,41 @@ describe('classic uses', () => { roulette.rotateCounterClockwise(2); expect(roulette.display()).toBe('Headphones'); // Headphones }); + + it('@example sliding window', () => { + // Maximum function of sliding window + function maxSlidingWindow(nums: number[], k: number): number[] { + const n = nums.length; + if (n * k === 0) return []; + + const deq = new Deque(); + const result: number[] = []; + + for (let i = 0; i < n; i++) { + // Delete indexes in the queue that are not within the window range + if (deq.length > 0 && deq.first! === i - k) { + deq.shift(); + } + + // Remove all indices less than the current value from the tail of the queue + while (deq.length > 0 && nums[deq.last!] < nums[i]) { + deq.pop(); + } + + // Add the current index to the end of the queue + deq.push(i); + + // Add the maximum value of the window to the results + if (i >= k - 1) { + result.push(nums[deq.first!]); + } + } + + return result; + } + + const nums = [1, 3, -1, -3, 5, 3, 6, 7]; + const k = 3; + expect(maxSlidingWindow(nums, k)).toEqual([3, 3, 5, 5, 6, 7]); // Output: [3, 3, 5, 5, 6, 7] + }); });