Skip to content

Dart solutions for the Grind 75 coding challenges. LeetCode, Blind 75, Neetcode 150, Grind 169. Data structures and algorithms.

Notifications You must be signed in to change notification settings

dartsidedev/grind75

Folders and files

NameName
Last commit message
Last commit date

Latest commit

Β 

History

82 Commits
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 

Repository files navigation

Grind 75 in Dart

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.

Resources

Important todos

  • All sorting algorithms. What's a good default? What is used by Dart (and other progr. languages)? Faster than nlogn?

Cheat sheet

Here, I summarize the core ideas, Dart syntax, data structures or algorithms that I keep forgetting.

Sorting

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 $O(n \log n)$ comparisons, in the worst case $O(n^2)$.

For lists with a length less than 32, Dart uses insertion sort, $O(n^2) $.

Queue

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 $O(1)$ $O(1)$ $O(1)$
ListQueue amortized $O(1)$ $O(1)$ $O(1)$

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.
}

Regex

// Assumes string's of length 1
bool get isAlphaNumericChar => RegExp(r'^[\w\d]$').hasMatch(this);
// or RegExp(r'^[a-zA-Z0-9]$')

ASCII code units

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;
}

Tree traversal

  • Preorder: root, left, right
  • Post order: left, right, root
  • Inorder: left, root, right

Binary search

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/

Reverse linked list

Must know it well!

Mix

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;
}

Disclaimers

Coding conventions

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 simple while, 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.

Test quality

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).

Solution descriptions

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.

Challenges

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

Solution LeetCode

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: $n$ is the number of elements in the list.

  • 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.
  • 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).
Valid Parentheses 🏷 stack

Solution LeetCode

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: $n$ is the length of the input string.

  • 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

Solution LeetCode

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, $n$ and $m$ are the lengths of the input linked lists:

  • 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

Solution LeetCode

Clarify at the interview:

  • What is the input, int vs num vs double? In this case, we work with int.
  • 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, $n$ is the length of the list:

  • 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)$.
Valid Palindrome 🏷 string

Solution LeetCode

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, $n$ is the length of the string:

  • 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

  1. Filter invalid characters and transform to lowercase (split+where+map+join).
  2. Check if palindrome
    • two pointers (now significantly easier to implement), or
    • reverse the string and compare against the original.

Complexity analysis, $n$ is the length of the string:

  • 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 ($O(n)$ vs. $O(1)$). In practice, it has both a worse memory usage and slower runtime.

Invert Binary Tree 🏷 binary tree

Solution LeetCode

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, $n$ is the number of the nodes in the tree:

  • 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

Solution LeetCode

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 $n$ is the length of the strings:

  • 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

Solution LeetCode

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

Iterative Solution in Dart LeetCode

Depth-first search, either iteratively or recursively.

Complexity analysis, where $n$ is the number of rows and $m$ in the number of columns in the image, and the total number of pixels is $p = m n$.

  • 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

Solution LeetCode

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 $n$ is the length of the list.

  • 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 and end indices, a third for calculating the sum.
    • Time complexity: $O(n^3)$ for the three nested loops.
    • Space complexity: $O(1)$.
  • Optimized brute force: One loop for moving the start index. The second loop moves the end index while simultaneously updating the currentMax value. Compare against max.
    • 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)$.
  • Divide and Conquer: TODO.
    • Time complexity: $O(n \log n)$.
    • Space complexity: $O(\log n)$.

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.

Solution LeetCode

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 $p$ and $q$, when one value is on the left side and the other value is on the right side of the node.

When both $p$ and $q$ are smaller than current, go to the left, when both $p$ and $q$ are larger, go to the right.

When one is smaller, the other one is larger, the current node is the lowest common ancestor. When the current value matches one of $p$ or $q$, the current node is the lowest common ancestor.

The solution can be done iteratively or recursively.

Complexity analysis, where $n$ is the number of nodes in the binary search tree, and $h$ is the height of the binary search tree.

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

Solution LeetCode

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

Solution LeetCode

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

Solution LeetCode

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

Solution LeetCode

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

Solution LeetCode

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...

Climbing Stairs 🏷 dynamic programming

Solution LeetCode

It's Fibonacci's twin sister.

Longest Palindrome 🏷 string

Solution LeetCode

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

Solution LeetCode

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

Solution LeetCode

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
Add Binary 🏷 binary

Solution LeetCode

Add Binary.

Middle of the Linked List 🏷 linked list

Solution in Dart (two pointers) LeetCode

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

Solution LeetCode

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

Solution LeetCode

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

Solution LeetCode

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

Solution LeetCode

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

Solution LeetCode

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

Solution LeetCode

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

Solution LeetCode

  • 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

Solution LeetCode

  • 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)
Number of 1 Bits 🏷 binary

Solution LeetCode

Go over the binary operators and you will be fine.

Maximum Depth of Binary Tree 🏷 binary tree

Solution LeetCode

Recursive, iterative solutions. Depth = max(left, right) + 1.

Diameter of a Binary Tree 🏷 binary tree

Solution LeetCode

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

Solution LeetCode

Solutions:

  • number of bits, n times...
  • TODO: understand and solve all the other solutions
Same tree 🏷 binary tree

Solution LeetCode

Solutions:

  • override ==
  • recursive
  • iterative (use stack)
Subtree of Another Tree 🏷 binary tree

Solution LeetCode

Solutions:

  • See same tree for comparing two trees. Traverse the tree and check each node.
Reverse Bits 🏷 binary

Solution LeetCode

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 and double) 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

Solution LeetCode

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. $n$ is the length of the input list. Time complexity: $O(n)$, as you iterate over the whole list in a single pass. Pushing to and popping off the stack is $O(1)$. Space complexity: $O(n)$, as the supporting stack can never be more than half the length of the input.

Sort Colors 🏷 array

Solution LeetCode

If we could use Dart's built-in sorting algorithm, it's a one-liner: dual pivot quicksort $O(n \log n)$ average case, $O(n^2)$ worst case.

We know that the possible values can only be 0, 1, 2, therefore, we can use counting sort.

Complexity analysis. $n$ is the length of the input list. Time complexity: $O(n)$, as you iterate over the whole list first to count, then to overwrite. Space complexity: $O(1)$, as we only need three supporting counter variables.

TODO: There is another one-pass solution I didn't check. Dutch flag.

Misc

Save your completed questions

console.log(JSON.stringify(JSON.parse(localStorage.getItem('1:completedQuestions')), null, 2));

About

Dart solutions for the Grind 75 coding challenges. LeetCode, Blind 75, Neetcode 150, Grind 169. Data structures and algorithms.

Topics

Resources

Stars

Watchers

Forks

Languages