<!--your preparation HTML code goes here-->
var myTasks = [];
myCopy = null;
function flattenTasks(tasks, flattenContacts = false){
let flattened = [];
tasks.forEach(task => {
if (flattenContacts) {
flattened.push(
{
task,
'contacts': task?.contacts?.length ? task.contacts.map(c => c.id) : [],
'subtasks': task?.subtasks?.length ? task.subtasks.map(s => s.id_uuid) : [],
}
)
} else {
flattened.push(
{
task,
'subtasks': task?.subtasks?.length ? task.subtasks.map(s => s.id_uuid) : [],
}
)
}
if (task?.subtasks?.length) {
flattened = [flattened, flattenTasks(task.subtasks, flattenContacts)];
}
})
return flattened;
}
myCopy = flattenTasks(myTasks);
function flattenTasks(tasks) {
const result = [];
const stack = [tasks];
while (stack.length) {
const task = stack.pop();
result.push({ task });
if (task.subtasks) {
stack.push(task.subtasks);
}
}
return result;
}
myCopy = flattenTasks(myTasks);
--enable-precise-memory-info
flag.
Test case name | Result |
---|---|
recursion | |
iterative |
Test name | Executions per second |
---|---|
recursion | 62710.5 Ops/sec |
iterative | 83154.1 Ops/sec |
The benchmark described tests the performance characteristics of two different approaches to flattening a potentially nested array of tasks: one using recursion and the other using an iterative approach (specifically, a stack-based method). The context provided, including names and script preparation code, guides us through how the benchmarks are set up and executed.
Recursive Approach
flattenTasks
) traverses the task structure depth-first. It creates an array (flattened
) where each task is pushed along with its subtasks, and if flattenContacts
is set to true
, it also extracts contact IDs.Iterative Approach
In this benchmark, no external libraries were used, and no special JavaScript features or syntax (like generators or promises) were leveraged. The code relies solely on basic JavaScript constructs (functions, arrays, objects) to implement both approaches. This accessibility makes it understandable to a wide range of software engineers.
The results of the benchmark indicate that the iterative approach performed significantly better than the recursive one, executing approximately 83,154 operations per second compared to around 62,710 for the recursive approach. This outcome is consistent with common expectations in JavaScript performance when dealing with large datasets and deep hierarchy structures.
There are other alternatives to the two methods discussed:
Using Libraries: Libraries like Lodash provide utility functions such as _.flattenDeep
. They are optimized and can handle deep flattening efficiently. However, they may add additional overhead due to the library size.
Functional Programming Approaches: Using functional programming techniques, such as higher-order functions (map
, reduce
), could also serve as a basis for flattening arrays, but they may not provide better performance due to the additional function calls and array manipulations.
Tail Recursion: Implementing a tail-recursive version of the recursive approach can potentially offer optimizations in environments that support tail call optimization (TCO), although TCO is not universally supported in JavaScript engines.
Web Workers: For very large data sets, utilizing web workers can help offload the processing to a background thread, thus keeping the user interface responsive. This is particularly useful when there's a possibility of long processing times due to deep or complex nesting.
Each of these alternatives has its trade-offs in terms of readability, maintainability, and performance that should be considered depending on the specific use case and environment.