All
*
the 169 challenges of the Grind 75 LeetCode study plan solved in beautiful, idiomatic Dart.
*
: Well, not yet, but I am on it! 🚀
You can find all the questions and solutions in the test
folder.
- All sorting algorithms. What's a good default? What is used by Dart (and other progr. languages)? Faster than nlogn?
Here, I summarize the core ideas, Dart syntax, data structures or algorithms that I keep forgetting.
You can find the default implementation of List.sort
in github.com/dart-lang
's sdk/lib/internal/sort.dart
.
It's the dual-pivot quicksort algorithm as presented in Vladimir Yaroslavskiy's paper. It uses two pivot elements (so partitions to three parts).
The dual pivot quicksort algorithm reduces the average number of swaps by 20%. The average number of comparisons stays the same.
On average, it takes
For lists with a length less than 32, Dart uses insertion sort,
A Queue
is a collection that can be manipulated at both ends.
Its implementers, DoubleLinkedQueue
and ListQueue
, are efficient for queue or stack usage.
// Queue, DoubleLinkedQueue, ListQueue
import 'dart:collection';
The DoubleLinkedQueue
is an implementation of the Queue
based on a double-linked list.
It allows constant time add, remove-at-ends and peek operations.
final doubleLinkedQueue = DoubleLinkedQueue<int>();
The ListQueue
is a list based Queue
.
It keeps a cyclic buffer of elements, and grows to a larger buffer when it fills up.
This guarantees constant time peek and remove operations, and amortized constant time add operations.
final initialCapacity = 16; // Default init capacity is 8.
final listQueue = ListQueue<int>(initialCapacity);
add (to ends) | remove (at ends) | peek | |
---|---|---|---|
DoubleLinkedQueue |
|||
ListQueue |
amortized |
Minimalist extensions for stacks and queues.
Sometimes, I find that the traditional queue operations come easier, especially if I am under stress and I feel pressured to be fast and clear while at an interview.
For this reason, I prefer to add the "minimalist" queue and stack static extension methods on the Queue
.
extension MinimalistQueue<E> on Queue<E> {
void enqueue(E value) => addLast(value);
E dequeue() => removeFirst(); // throws StateError if empty!
E? peek() => firstOrNull; // For most problems, peek is not needed.
}
extension MinimalistStack<E> on Queue<E> {
void push(E value) => addLast(value);
E pop() => removeLast(); // throws StateError if empty!
E? peek() => lastOrNull; // For most problems, peek is not needed.
}
// Assumes string's of length 1
bool get isAlphaNumericChar => RegExp(r'^[\w\d]$').hasMatch(this);
// or RegExp(r'^[a-zA-Z0-9]$')
For day-to-day work, knowing and remembering the ASCII code units of different characters is completely useless.
However, in many coding challenges, it can come in handy.
starts | ends | count | |
---|---|---|---|
numbers | 48 (0 ) |
57 (9 ) |
10 |
uppercase | 65 (A ) |
90 (Z ) |
26 |
lowercase | 97 (a ) |
122 (z ) |
26 |
For example, if you know that the input is ASCII-only, you can determine very easily whether a character is alphanumeric using code units.
// Assumes string's of length 1
bool get isAlphaNumericChar {
final codeUnit = codeUnitAt(0);
final isNumber = 48 <= codeUnit && codeUnit <= 57;
final isUpperCase = 65 <= codeUnit && codeUnit <= 90;
final isLowerCase = 97 <= codeUnit && codeUnit <= 122;
return isNumber || isLowerCase || isUpperCase;
}
In this example, let's say we know about the input string that it only contains lowercase letters (English alphabet). If we need to store, for example, a counter for each letter in a list, we can write this static extension getter to convert from strings to the indices for the list.
extension CharToIndex on String {
// Assumes only lowercase and length == 1
int get charIndex => codeUnitAt(0) - 97;
}
- Preorder: root, left, right
- Post order: left, right, root
- Inorder: left, root, right
You need to be able to write a binary search in a minute, you don't have 10 minutes to think about edge cases.
TODO: Check card: https://leetcode.com/explore/learn/card/binary-search/
Must know it well!
import 'dart:math';
final maxValue = max(a, b);
// or
import 'dart:math' as math;
final maxValue = math.max(a, b);
import 'package:collection/collection.dart';
/// Equality on lists.
///
/// Two lists are equal if they have the same length and their elements
/// at each index are equal.
bool match(List<int> a, List<int> b) => const ListEquality().equals(a, b);
// part of dart core
// Check whether two references are to the same object.
external bool identical(Object? a, Object? b);
// A simplified (int only) version of
// the ListEquality.equals from package:collection
bool listEquals(List<int>? list1, List<int>? list2) {
// Are they identical?
if (identical(list1, list2)) return true;
// Consider nulls
if (list1 == null || list2 == null) return false;
// Check lengths and return early if don't match
var length = list1.length;
if (length != list2.length) return false;
// Check every item, if they don't match return
for (var i = 0; i < length; i++) {
if (list1[i] != list2[i]) return false;
}
// Yay, they are equal!
return true;
}
I do not always follow the official Dart style guide or rules that are in most popular linting libraries enabled.
You can find below the list of rules that I do not follow with a reason as to why that is.
curly_braces_in_flow_control_structures
: I don't want to "spend" three lines just to write a simplewhile
,if
,for
.avoid_multiple_declarations_per_line
: sometimes two declarations just belong together, and at interviews, you really don't have enough time to declare everything in a new line.
I didn't spend time writing the perfect test description for my tests. I made sure that I have enough tests to spot any potential mistakes and that I add the examples from LeetCode.
This doesn't mean that in your day job, you should write the tests like I did in this repository, it just means that I wanted to focus on solving the problem correctly.
Unfortunately, LeetCode doesn't work with Dart, if you want to change that, upvote this issue. LeetCode doesn't do it for me for Dart).
The solution descriptions are mostly for myself. For cases, where the challenge was very easy for me, I didn't spend much effort on describing the solution.
Below you can find a list of challenges that I already solved.
Following Neetcode's advice, I summarize each challenge's core concepts and different ways to solve the challenges with its pros and cons.
Two Sum 🏷 array
Exactly one solution. You may not use the same element twice. Return indices of the two numbers such that they add up to target.
We need to return the indices. Create a map where the key is the number value and the value is its index.
Iterate over numbers: store visited numbers and their indices in the map.
As you iterate, check whether current number has a complement in the map that adds up to target. If it does, return the indices. If it doesn't, add number to the map.
Complexity analysis:
-
Time complexity:
$O(n)$ , as you might iterate over the whole list. -
Space complexity:
$O(n)$ , as you might need to store almost all elements and their indices in the map.
Other solutions:
-
Brute force: double loop, return when the two numbers add up to the target.
-
Time complexity:
$O(n^2)$ , for each of$n$ elements, we try to find its complement by looping through the rest of the list. -
Space complexity:
$O(1)$ , no extra space that depends on the input size is necessary.
-
Time complexity:
-
Sort list first, then on a sorted list, find values
with two pointers (from start and end).
-
Time complexity:
$O(n \log n)$ , because sort$O(n \log n)$ + two pointers$O(n)$ . -
Space complexity:
$O(n)$ , sort and two-pointers would be possible with$O(1)$ , but we need to store original indices$O(n)$ . - This solution, as the list is not already sorted and we need to return the original indices, is fairly complicated, and doesn't perform too well (neither in space nor time complexity).
-
Time complexity:
Valid Parentheses 🏷 stack
input string consists of parentheses/brackets only
()[]{}
Push opening parenthesis/bracket to a stack. Pop off when closing, and make sure they are matching. Don't forget to check at the end if the stack is empty. Remember to pop off only if stack is not empty (or use peek).
Complexity analysis:
-
Time complexity:
$O(n)$ , as you iterate over the whole input string, and potentially adding half of them to a stack. Pushing and popping on a stack can be a$O(1)$ operation if the stack is efficient. -
Space complexity:
$O(n)$ , we need a stack that might contain$n$ elements.
Solution variant, you could immediately push the closing parenthesis onto the stack, it actually ends up looking pretty nice.
Merge Two Sorted Lists 🏷 linked list
Trick: pre-head pointer significantly simplifies the iterative algorithm.
While both lists are not empty, pick one off the lists and add to the results. Move pointer. Do not forget to add the remaining items of the longer list to the list. Return the pre-head's next as result.
Consider empty nodes. Do not assume the two lists are of equal length.
Complexity analysis,
-
Time complexity:
$O(n+m)$ (linear with total length), because we simply iterate over both lists and picking the correct. -
Space complexity:
$O(1)$ , only need a couple of variables, constant space, and the returned list consists of nodes already created before the solution algorithm ran.
Alternative solutions:
- Recursive solution. Basically the same, just with different space complexity due to the stack.
Best Time to Buy and Sell Stock 🏷 array
Clarify at the interview:
- What is the input,
int
vsnum
vsdouble
? In this case, we work withint
. - Can it be negative/positive? It doesn't affect the algorithm.
- What should the solution return if it is impossible to achieve any profit? Return 0 if no profit possible.
Keep track of min price "so far". Current profit is price minus the min price so far. Update max profit if current profit greater. Handle negative profit edge case (must return 0).
Complexity analysis,
-
Time complexity:
$O(n)$ , as you iterate over the whole list in a single pass. -
Space complexity:
$O(1)$ , because you only need two variables holding integers.
Alternative solutions:
- brute force: double loop, calculate profit for each possible pair.
-
Time complexity:
$O(n^2)$ , for each number, sweep the rest of the list. -
Space complexity:
$O(1)$ .
-
Time complexity:
Valid Palindrome 🏷 string
Start with two pointers at each end of the string. If a letter is not alphanumeric, move pointer to next alphanumeric. Make sure that the left pointer is always to the left of the right pointer. Whenever the two pointers contain alphanumeric chars, compare. If the values for the two pointers don't match, return "not a palindrome".
Complexity analysis,
-
Time complexity:
$O(n)$ , as we traverse over each character at most once. -
Space complexity:
$O(1)$ , because you only need two variables holding integers.
Alternative solutions:
Functional approach
- Filter invalid characters and transform to lowercase (
split
+where
+map
+join
). - Check if palindrome
- two pointers (now significantly easier to implement), or
- reverse the string and compare against the original.
Complexity analysis,
-
Time complexity:
$O(n)$ , as we iterate over the string: filter out non-alphanumeric values, map to lowercase, create string, create reversed, compare. All of these steps run in linear time. -
Space complexity:
$O(n)$ , for the supporting data structures: split list, joined string, reversed joined.
The functional approach, compared to the optimal iterative approach, has worse space complexity (
Invert Binary Tree 🏷 binary tree
The inverse of a tree with root
$R$ , and subtrees$r$ and$l$ , is a tree with root$R$ , whose right subtree is the inverse of$l$ , and whose left subtree is the inverse of$r$ .
Invert tree recursively. Swap children, then invert left and right subtrees. Handle null case.
Both pre-order and post-order traversal give the right answer.
Complexity analysis,
-
Time complexity:
$O(n)$ , we visit each node once. -
Space complexity:
$O(n)$ , the recursion stack will be as high as the tree, which in the worst case is equal to the number of nodes.
TODO: Solve iteratively, both post and preorder.
Valid Anagram 🏷 string
An Anagram is a word or phrase formed by rearranging the letters of a different word or phrase, typically using all the original letters exactly once.
Two words are anagram if they both contain the same letters, the same number of times.
Return early if lengths don't match. Create counter (list or map). Check if counts match.
Complexity analysis, where
- We assume
$n$ is the length of both strings. If the lengths don't match, we can return immediately. -
Time complexity:
$O(n)$ as we need to iterate over the strings to create the frequency/counter table. -
Space complexity:
$O(1)$ , the counter's space requirements stay constant no matter how large$n$ is: The counter will be a fixed length list with length 26. The same is true even for unicode, just the map will be much bigger.
Consider creating a helper class Counter
that can be created
from a string and then compared against another counter.
The Counter
should wrap a fixed-length list for ASCII-only.
If you are not allowed to use package:collection
's ListEquality
, be prepared to implement it.
Follow up: unicode? In Version 14.0 of the Unicode Standard, there are 144K characters. To fit all possible unicode characters in a list, the list would have to contain 1M+ elements (unicode is evolving, 1M+ is the hypothetical max). That's wasteful if the strings are short (and anything less than a million characters counts as short in this case). Change from fixed length list to hash map for storing the counters.
Binary Search 🏷 binary search
Learn to write binary search, it doesn't really get much simpler than that. Pay attention to empty list, first element, last element. Practice templates.
Flood Fill 🏷 graph
Depth-first search, either iteratively or recursively.
Complexity analysis, where
-
Time complexity:
$O(n m) = O(p)$ as we might process every pixel. -
Space complexity:
$O(n m) = O(p)$ for the stack (either the call stack in the recursive solution or the queue/stack for the iterative solution).
Maximum Subarray 🏷 dynamic programming
Iterate over the list. Keep track of the "current" maximum sum, and the maximum sum "so far".
When the current single item is better than the current sum plus the current single item, reset the current max sum to the current item only. If it isn't better, add to the sum.
This basically means that if the current sum is negative, you stop using the past values.
current = math.max(current + v, v);
// is the same as:
current = current.isNegative
?
value : current + value;
Don't forget to update the maximum value "so far" when needed.
PS: This is Kadane's algorithm.
Complexity analysis, where
-
Time complexity:
$O(n)$ as we need to iterate over all items once. -
Space complexity:
$O(1)$ as we only need to use two variables, one for the "current" max and the other for the max "so far". This space doesn't depend on$n$ .
Alternative solutions:
-
Brute force: Three loops. Two for moving
start
andend
indices, a third for calculating the sum.-
Time complexity:
$O(n^3)$ for the three nested loops. -
Space complexity:
$O(1)$ .
-
Time complexity:
-
Optimized brute force: One loop for moving the
start
index. The second loop moves theend
index while simultaneously updating thecurrentMax
value. Compare againstmax
.-
Time complexity:
$O(n^2)$ for the two nested loops. The improvement over the$O(n^3$ solution comes from the fact that we recognized that when we add one more item to the sub-list, we don't need to recalculate the current sum completely: $ \sum_{k = 0}^{n + 1} a_k = a_{n + 1} \sum_{k = 0}^{n} a_k $. -
Space complexity:
$O(1)$ .
-
Time complexity:
-
Divide and Conquer: TODO.
-
Time complexity:
$O(n \log n)$ . -
Space complexity:
$O(\log n)$ .
-
Time complexity:
You can find out more about this issue on Wikipedia - Maximum Subarray Problem.
Lowest Common Ancestor of a Binary Search Tree 🏷 binary search tree
🐌 Can't submit solutions with Dart on LeetCode yet for this problem.
The lowest common ancestor is defined between two nodes
$p$ and$q$ as the lowest node in$T$ tree that has both$p$ and$q$ as descendants (where we allow a node to be a descendant of itself).
We need to realize that in a binary search tree, a node is the lowest common ancestor of
When both
When one is smaller, the other one is larger, the current node is the lowest common ancestor.
When the current value matches one of
The solution can be done iteratively or recursively.
Complexity analysis, where
Iterative solution:
-
Time complexity:
$O(h)$ . In the worst case,$h = n$ , and therefore$O(n)$ because if the binary search tree is unbalanced (or degenerated to a linked list) and the values are at the bottom, (almost) all nodes in the binary search tree will be visited. In a balanced binary search tree$h = \log n$ . In this case, the time complexity will be$O(\log n)$ . -
Space complexity:
$O(1)$ , no extra space needed apart from a variable.
Recursive solution:
-
Time complexity:
$O(n)$ (really $O(h)$), for balanced$O(\log n)$ . -
Space complexity:
$O(n)$ (as in the worst case$h = n$ ) as we need space for the recursion stack. In case the tree is balanced,$h = \log n$ .
Balanced Binary Tree 🏷 binary tree
Solutions. Iterative, recursive.
a binary tree in which the left and right subtrees of every node differ in height by no more than 1.
Get the height of the left and right subtrees. Use -1 to show that it is not balanced.
TODO: Iterative
Implement Queue using Stacks 🏷 stack
Have two stacks, and a phase internally whether we were pushing or popping off the queue. When changing phases, move all items from one stack to another first, then add or remove.
Linked List Cycle 🏷 linked list
Solutions:
- Iterate over linked list and store seen nodes in a map. If node is already seen, return that it has a cycle. If reaches the end of the list, it has no cycles. TC O(n), SC O(n)
- Double pointers. Fast and slow pointers. Fast moves two at a time, slow moves one at a time. if they "meet", it's a cycle. Careful with the stop condition.
First bad version 🏷 binary search
Learn to write binary search, it doesn't really get much simpler than that. Pay attention to empty list, first element, last element. Practice templates.
Ransom Note 🏷 hash table
For letter frequency counter, use map for unicode, use list of length 26 for ASCII lowercase.
Simple solution: You can use two maps/lists as frequency counters, then check whether the magazine's counters are highers for each position than in the ransom note.
Alternatives. Terminate early (earlier?): Start with one map for the magazine, then start removing letters based on the ransom note. Whenever your magazines hit 0 counts, return false. Alternative II. You can also sort and compare, sort and put it into a stack, etc...
Longest Palindrome 🏷 string
Build letter counter (frequency). Know your code units: A for 65, Z for 90, a for 97, z for 122. Alternatively, use a map.
We can use letters in pairs to build the palindrome length (use int division by 2). Then, if there was an odd letter, we can add it to the middle (counts as +1).
Reverse Linked List 🏷 linked list
Helpful: pre-head! TODO: check my past solutions, one of them must be intuitive enough to remember and solve in 3 minutes.
Majority Element 🏷 array
Potential solutions
- Double loop: Count how often the element is in the list, if it's greater than half, it's the solution. TC O(n^2), SC: O(1).
- Sort, then sweep for greatest. Sort TC O(n log n), SC O(1). Sweep TC O(n), SC O(1). Total: TC O(n log n), SC O(1).
- Improvement: once sorted, we don't need to sweep. It's always the middle element.
- Store counter in map, if majority, return. TC O(n), SC O(n).
- Random: Pick an index randomly, there is at least a 50% chance it will be the majority item. O(n) to verify. Repeat until found. Worst case scenario: infty.
TODO:
- Boyer-Moore
- Divide and Conquer
Middle of the Linked List 🏷 linked list
Potential solutions:
- First pass: count elements. Second pass: go to the middle. TC O(n), SC O(1)
- One-pass algorithm: two pointers, slow and fast. When fast is at the end, slow is in the middle.
Contains duplicate 🏷 array
Potential solutions:
- brute force: double loop. TC O(n^2), SC O(1)
- sort then look for duplicates. TC O(n log n + n), SC O(1). Mutates input! (or if it doesn't mutate, then it needs a copy O(n))
- store seen in map, iterate over elements and check if already seen. TC O(n), SC O(n). Does not mutate input.
Roman to Integer 🏷 math
Keep track of value so far. Iterate over the string. Check if the next two characters make up an exception. If they do: add to value, skip next char. If they don't: treat first char as regular, add to value.
Alternative solutions: Left to Right . Right to Left.
Single Number 🏷 binary
every element appears twice except for one
Solutions:
- Iter over items: if in second array, remove the item, if it isn't, add the item. In the end, only the single number will stay
- Build map of frequencies. Iterate over map entries, find where value is 1, return key. TC O(n), SC O(n) (this complexity is not accepted according to the answer)
- "Bit xor"
^
all the way. Can reduce or loop. - math: 2 * (a1 + a2 + ... + an + b) - (a1 + a1 + ... + an + an + b) = b. Sum up all items in list. Then add all items in a set, sum up, double it. The diff is the number.
Missing Number 🏷 binary
Solutions:
- Sort TC O(n log n), then find missing TC O(n).
- Bit xor again!
^
. Xor together all the numbers, then xor with n. The result is the missing number. - Calculate expected sum, calculate actual sum, the diff is the missing number
- expected sum can be either calculated O(n), or use Gauss formula
Palindrome Number 🏷 math
First approach: convert to string, then solve it as if was a string.
Second approach: convert it to a list, then solve it as if it were a list. Use %
and ~/
.
Third approach: get the last and first digits. Transform the input. Repeat. Return false if they don't match.
Fourth approach: create reverted number, then compare integers.
Squares of a Sorted Array 🏷 array
Must consider possible negative numbers!
Very brute force solution (does not mutate input): Map square (TC: O(n), SC: O(n)), sort (TC: O(n log n), SC: O(1)).
Brute force solution (mutates input, no extra space at all): Square each number in place (TC: O(n), SC: O(1)), sort the list (TC: O(n log n), SC: O(1)).
Two-pointer solution: from one of the two ends of the list will come the next biggest square (postive, negative numbers) . Move two pointers, and the biggest square will be added to a list. You can create a fixed size list beforehand, in this case, the list will be filled from the end (largest) to the start (smallest). Complexity: O(n), space O(1) (if we count the output: O(n)).
K Closest Points to Origin 🏷 heap
- Helpful math knowledge: do not need sqrt, just use "x * x + y * y"
- Align with interviewer: could the return value be an Iterable?
Possible solutions:
- Total brute force: double loop: when smallest item found, remove it from list, copy into results.
- Sort list by "square sums" TC O(n log n), SC O(1). Copy first k elements into a list TC O(k), SC O(k)
- Add all elements into a min heap TC O(n), SC O(n). "Pop off" the smallest k elements TC O(k log n) (result could be an iterable, not necessarily a list)
- Add k elements into a max heap. After the kth, every time you add something into the heap, pop off the largest value. At the end, add remaining items to the result (either as list or iterable).
- TODO: Quick Select
Backspace String Compare 🏷 stack
- build two stacks based on inputs: if
#
, pop off the stack. Then, compare the two stacks: check length first, then pop the items of and if there is no match, return false. TC O(n), SC O(n) - iterate from the end. If
#
, continue. Char-generator function, for both, get next, compare. TC O(n), SC O(1)
Maximum Depth of Binary Tree 🏷 binary tree
Recursive, iterative solutions. Depth = max(left, right) + 1.
Diameter of a Binary Tree 🏷 binary tree
Calculate height for each node: height = max(left, right) + 1. Calculate diameter for each node: diameter = leftHeight + rightHeight. While calculating the heights and diameters, update "max diameter".
Do it recursively, and iteratively.
Iterative postorder traversal is easy with two stack (though also possible with one), and you keep track of heights in map of nodes to heights.
Counting bits 🏷 binary
Solutions:
- number of bits, n times...
- TODO: understand and solve all the other solutions
Subtree of Another Tree 🏷 binary tree
Solutions:
- See same tree for comparing two trees. Traverse the tree and check each node.
Reverse Bits 🏷 binary
Get rightmost bit of the value and add it to the result. Then shift the input value to the right and shift the result to the left.
Reminder: know your bitwise operators: AND, OR, XOR, NOT, left and right shifts.
According to Numbers in Dart:
Depending on the platform, those numeric types (
int
anddouble
) have different, hidden implementations. In particular, Dart has two very different types of targets it compiles to: native (most often, a 64-bit mobile or desktop processor) and web (JavaScript as the primary execution engine).
Keep in mind that on JavaScript, that "JavaScript converts numbers into 32 bits before the implementation of bitwise operators" which can lead to surprising behavior.
Evaluate Reverse Polish Notation 🏷 stack
Iterate over input. If it's a number, add to stack. If it's an operation, perform operation on top 2 items in the stack, and push result back to stack.
Complexity analysis.
Sort Colors 🏷 array
If we could use Dart's built-in sorting algorithm, it's a one-liner: dual pivot quicksort
We know that the possible values can only be 0, 1, 2, therefore, we can use counting sort.
Complexity analysis.
TODO: There is another one-pass solution I didn't check. Dutch flag.
console.log(JSON.stringify(JSON.parse(localStorage.getItem('1:completedQuestions')), null, 2));