Saturday, December 28, 2019

Leetcode solution 59: Sprial Matrix II


Problem Statement 


Given a positive integer n, generate a square matrix filled with elements from 1 to n2 in spiral order.
Example:
Input: 3
Output:
[
 [ 1, 2, 3 ],
 [ 8, 9, 4 ],
 [ 7, 6, 5 ]
]
 Problem link

Video Tutorial

You can find the detailed video tutorial here

Thought Process

Straight forward question, could use Spiral Matrix exact same recursion algorithm to solve this, you can also watch that video tutorial here

Still needs pay attribute to this guy, had another great solution:
https://leetcode.com/problems/spiral-matrix-ii/discuss/22282/4-9-lines-Python-solutions

Solutions

Simulation using Recursion

public int[][] generateMatrix(int n) {
assert n >= 0;
int[][] res = new int[n][n];
this.generateMatrixHelper(0, 0, n, n, res, 1);
return res;
}
private void generateMatrixHelper(
int i,
int j,
int rowSize,
int colSize,
int[][] matrix,
int num
) {
if (rowSize <= 0 || colSize <= 0) {
return;
}
if (rowSize == 1 && colSize == 1) {
matrix[i][j] = num;
return;
}
if (rowSize == 1) {
for (int k = j; k < j + colSize; k++) {
matrix[i][k] = num;
num++;
}
return;
}
if (colSize == 1) {
for (int k = i; k < i + rowSize; k++) {
matrix[k][j] = num;
num++;
}
return;
}
// do the spiral
for (int k = j; k < j + colSize; k++) {
matrix[i][k] = num++;
}
for (int k = i + 1; k < i + rowSize; k++) {
matrix[k][j + colSize - 1] = num++;
}
for (int k = j + colSize - 2; k >= i; k--) {
matrix[i + rowSize - 1][k] = num++;
}
for (int k = i + rowSize - 2; k > i; k--) { // both the start and end need to be i, j, and also care about length
matrix[k][j] = num++;
}
this.generateMatrixHelper(i + 1, j + 1, rowSize - 2, colSize - 2, matrix, num);
}
Time Complexity: O(M*N) where M, N is row and col of matrix
Space Complexity: O(M*N) since we used list to store the result, where M, N is row and col of matrix


References

Friday, December 27, 2019

System design interview: how to design a feeds system (e.g., Twitter, Instagram and Facebook news feed)

System design interview: how to design a feeds system (e.g., Twitter, Instagram and Facebook news feed)

Methodology: READ MF!

Please use this "READ MF!" framework for software engineer system interview purpose.

Key designs and terms 




  • So far the best detailed explanation on designing twitter is from Raffi (used to be VP of Twitter) on QCon. Presentation, Slides [All credits go to QCon). Raffi is very smart and articulate, really solid guy!
  • Read heavy system, not write heavy, optimized for user timeline. To be clear, there are two timelines, one is user's own tweets (easy to do), the other is the main timeline which is an aggregation of all the tweets from the people that user follows. 
  • Pre-calculate all the timelines. This is the interesting part of the design vs using a mysql and having index to query in realtime, which would not be scalable. When a tweet is posted, the tweets service would
    • Store this tweet in memory, and that later would be flushed to a main DB
    • Call the fanout deliver service to publish this tweet to all the users' timeline that followed this particular user. It could simply store a tweet ID (and later the content could be retrieved from the Tweets Cache) or hydrating the entire text content is also fine (note how we want to handle eidt or delete, twitter probably doesn't allow delete)
    • Call search service to index (Lucene). The search index is also hosted in memory on Redis. Note search here needs to fanout to all search clusters but due to the in memory hosting, it's acceptable.
  • Always remember disk access is at least 100k times slower than memory access, e.g, disk is 10ms vs 100ns on memory. https://gist.github.com/jboner/2841832
  • With the pre-caculate timeline design, there might be race conditions when celebrity (people with millions of followers) starts to talking to each other with replies. E.g., celebrity A tweets something, takes 30 sec to deliver to all the followers,  celebrity B replies before deliver finishes, some followers follow both A and B might see B's reply first before A's original post. One cheat workaround is to sort by timestamp or tweet IDs, but they are also experimenting with only pre-calculate non-celebrity tweets, and when generating timeline, realtime fetching the celebrity tweets. It depends which way is better and also the user experience. This is a good stop point to talk to your interviewers in real world about tradeoffs.
  • Since twitter is heavily relying on cache, you might want to checkout how they optimized the caching with twemproxy.

Baozi Youtube Video

Saturday, December 21, 2019

Leetcode solution 54: Sprial Matrix


Problem Statement 

Given a matrix of m x n elements (m rows, n columns), return all elements of the matrix in spiral order.
Example 1:
Input:
[
 [ 1, 2, 3 ],
 [ 4, 5, 6 ],
 [ 7, 8, 9 ]
]
Output: [1,2,3,6,9,8,7,4,5]
Example 2:
Input:
[
  [1, 2, 3, 4],
  [5, 6, 7, 8],
  [9,10,11,12]
]
Output: [1,2,3,4,8,12,11,10,9,5,6,7]
 Problem link

Video Tutorial

You can find the detailed video tutorial here

Thought Process

It's a straight forward implementation question, we can simply just simulate the spiral order and print. You can choose to do it either iteratively (see reference to download the official Leetcode solution) or use recursion.

There is this awesome one line solution from this guy which is pretty insane.
def spiralOrder(self, matrix):
    return matrix and list(matrix.pop(0)) + self.spiralOrder(zip(*matrix)[::-1])
https://leetcode.com/problems/spiral-matrix/discuss/20571/1-liner-in-Python-%2B-Ruby

Solutions

Simulation using Recursion

public List<Integer> spiralOrder(int[][] matrix) {
List<Integer> res = new ArrayList<>();
if (matrix == null || matrix.length == 0 || matrix[0].length == 0) {
return res;
}
this.printSpiralOrder(0, 0, matrix.length, matrix[0].length, res, matrix);
return res;
}
// printSpiralOrder is a poor name, function starts with verb, printSpiralOrder, class is noun, function is verb
private void printSpiralOrder(int i, int j, int rowSize, int colSize, List<Integer> res, int[][] matrix) {
if (rowSize <= 0 || colSize <= 0) {
return;
}
if (rowSize == 1 && colSize == 1) {
res.add(matrix[i][j]);
return;
}
if (rowSize == 1) {
for (int k = j; k < j + colSize; k++) {
res.add(matrix[i][k]);
}
return;
}
if (colSize == 1) {
for (int k = i; k < i + rowSize; k++) {
res.add(matrix[k][j]);
}
return;
}
// do the spiral
for (int k = j; k < j + colSize; k++) {
res.add(matrix[i][k]);
}
for (int k = i + 1; k < i + rowSize; k++) {
res.add(matrix[k][j + colSize - 1]);
}
for (int k = j + colSize - 2; k >= i; k--) {
res.add(matrix[i + rowSize - 1][k]);
}
for (int k = i + rowSize - 2; k > i; k--) { // both the start and end need to be i, j, and also care about length
res.add(matrix[k][j]);
}
this.printSpiralOrder(i + 1, j + 1, rowSize - 2, colSize - 2, res, matrix);
}
Time Complexity: O(M*N) where M, N is row and col of matrix
Space Complexity: O(M*N) since we used list to store the result, where M, N is row and col of matrix


References

Saturday, December 14, 2019

Leetcode solution 322: Coin Change


Problem Statement 

You are given coins of different denominations and a total amount of money amount. Write a function to compute the fewest number of coins that you need to make up that amount. If that amount of money cannot be made up by any combination of the coins, return -1. Example 1:
Input: coins = [1, 2, 5], amount = 11
Output: 3 
Explanation: 11 = 5 + 5 + 1
Example 2:
Input: coins = [2], amount = 3
Output: -1
 
Problem link

Video Tutorial

You can find the detailed video tutorial here

Thought Process

This is a classic problem and greedy might be the first thing comes into mind. For example, sort the coins on denomination and always use the largest amount coin. However, the optimal solution might not include the largest denomination.

for example , [1, 3, 4] and target value 6. The optimal is two 3s, not [4, 1, 1].

It seems we have to list out all the different combinations and find the min value, that leads to a brute fore solution as described here using recursion. It would be an exponential time complexity.

We can use a hash map as a lookup to remember for each amount, what would be the min cost. That would significantly reduce the time complexity from exponential to O(S*N) where S is the amount and N is the coins array size. It's not obvious why but once you see the DP solution, you can understand better, it's the same time complexity.

Another way is to use dynamic programming because the problem is asking for a single yes/no, extreme value(min, max), building up a lookup table using formula is the coding pattern. We have similar problems, Wild Card Matching, Regular Expression Matching

The idea is to build a lookup table for each amount, what's the minimal number of coins needed given current  denomination.
lookup[i] = min(lookup[i], lookup[i - coin value] + 1) given i - coin value >= 0

Solutions

Dynamic Programming

public int coinChangeDP(int[] coins, int amount) {
if (coins == null || coins.length == 0) {
return -1;
}
// lookup contains the min number of coins needed to reach the amount(i.e., index)
int[] lookup = new int[amount + 1];
lookup[0] = 0;
for (int i = 1; i <= amount; i++) {
int min = Integer.MAX_VALUE;
for (int coin : coins) {
if (i >= coin && lookup[i - coin] != -1) {
min = Math.min(min, lookup[i - coin] + 1);
}
}
lookup[i] = (min == Integer.MAX_VALUE ? -1 : min);
}
return lookup[amount];
}
view raw CoinChange.java hosted with ❤ by GitHub
Time Complexity: O(S*N) where S is the amount and N is the coins array size
Space Complexity: O(N) since we used a lookup array


References

Saturday, December 7, 2019

Leetcode solution 274: H-Index

包子小道消息@12/07/2019


从growth at all cost 到profitability mindset,都是被Uber Wework ​弄得。软银老板Masa的头发估计又得脱落几根,不过airbnb也许是2020年被大家追捧的一个新星,从cash at hand / total funds raised 的比例来看,就是下一个google,Facebook和zoom的节奏啊!

 

Problem Statement 

Given an array of citations (each citation is a non-negative integer) of a researcher, write a function to compute the researcher's h-index.
According to the definition of h-index on Wikipedia: "A scientist has index h if h of his/her N papers have at least h citations each, and the other N − h papers have no more than h citations each."
Example:
Input: citations = [3,0,6,1,5]
Output: 3 
Explanation: [3,0,6,1,5] means the researcher has 5 papers in total and each of them had 
             received 3, 0, 6, 1, 5 citations respectively. 
             Since the researcher has 3 papers with at least 3 citations each and the remaining 
             two with no more than 3 citations each, her h-index is 3.
Note: If there are several possible values for h, the maximum one is taken as the h-index.
Hint 1
An easy approach is to sort the array first.
Hint 2
What are the possible values of h-index?

Hint 3
A faster approach is to use extra space. 


Problem link

Video Tutorial

You can find the detailed video tutorial here

Thought Process

It’s not the most straightforward question to understand I have to admit that. I normally start with small examples to help me understand

The requirement is: his/her N papers have at least h citations each, and the other N − h papers have no more than h citations each

What happens if citation contains only 1 element?
  • Citations = [100] , N = 1, h = 1, N-h = 0
This gives us hint that h has an upper bound of N

What happens if citation contains only 2 elements? And we can assign some large citation values
  • Citations = [1000, 2000] , N = 2, h = 2. It seems h-index has not too much to do with the citation values itself, rather it’s about the index and the values.

We should now have a super brute force way, since h is bounded to be [0, N], we just go through each value and try to see if it could meet the h criteria (e.g., we start from N, then we see if there is at least N papers that have at least N citations) This would result in a O(N^2) time complexity.

Naturally, we normally can think of if we can sort the array to reduce the time complexity. After sort, using the below graph, the problem becomes find the citation more than the citation array index, and that citation index could be the h-index (that's why it's called h-index, not h-value, it's about index). This would give a O(NlgN) time complexity. Code is attached in the References section. 
source: leetcode solution

Finally, we can think about bucket sort since h-index is bounded from [0, N], and if paper citations are larger than N, it's just considered on counting on index N.

You know how to do a bucket sort 😄

Solutions

Use bucket sort

public int hIndex(int[] citations) {
if (citations == null || citations.length == 0) {
return 0;
}
// because we need from 0 to citations.length, h-index in [0, citations_length]
int[] counts = new int[citations.length + 1];
for (int i = 0; i < citations.length; i++) {
/*
if (citations[i] > citations.length) {
counts[citations.length] += 1;
} else {
counts[citations[i]] += Math.min(1, citations[i]); // this is for 0 citation case
}
this 4 lines translate into below line, sounds niubility!
*/
counts[Math.min(citations[i], citations.length)]++;
}
int citationCount = 0;
// Utilities.printIntArray(counts);
for (int i = counts.length - 1; i >= 0; i--) {
citationCount += counts[i];
if (i <= citationCount) {
return i;
}
}
// there is one and only one h-index, so this would never happen
return -1;
}
view raw HIndex.java hosted with ❤ by GitHub
Time Complexity: O(N) since we went through citations only once
Space Complexity: O(N) since we are using an extra array to keep track of the count


References

Saturday, November 30, 2019

Leetcode solution 133: Clone Graph

Problem Statement 

Given a reference of a node in a connected undirected graph.

Return a deep copy (clone) of the graph.

Each node in the graph contains a val (int) and a list (List[Node]) of its neighbors.

class Node {
    public int val;
    public List<Node> neighbors;
}

 

Test case format:

For simplicity sake, each node's value is the same as the node's index (1-indexed). For example, the first node with val = 1, the second node with val = 2, and so on. The graph is represented in the test case using an adjacency list.

Adjacency list is a collection of unordered lists used to represent a finite graph. Each list describes the set of neighbors of a node in the graph.

The given node will always be the first node with val = 1. You must return the copy of the given node as a reference to the cloned graph.

 

Example 1:

Input: adjList = [[2,4],[1,3],[2,4],[1,3]]
Output: [[2,4],[1,3],[2,4],[1,3]]
Explanation: There are 4 nodes in the graph.
1st node (val = 1)'s neighbors are 2nd node (val = 2) and 4th node (val = 4).
2nd node (val = 2)'s neighbors are 1st node (val = 1) and 3rd node (val = 3).
3rd node (val = 3)'s neighbors are 2nd node (val = 2) and 4th node (val = 4).
4th node (val = 4)'s neighbors are 1st node (val = 1) and 3rd node (val = 3).

Example 2:

Input: adjList = [[]]
Output: [[]]
Explanation: Note that the input contains one empty list. The graph consists of only one node with val = 1 and it does not have any neighbors.

Example 3:

Input: adjList = []
Output: []
Explanation: This an empty graph, it does not have any nodes.

Example 4:

Input: adjList = [[2],[1]]
Output: [[2],[1]]

 

Constraints:

  • 1 <= Node.val <= 100
  • Node.val is unique for each node.
  • Number of Nodes will not exceed 100.
  • There is no repeated edges and no self-loops in the graph.
  • The Graph is connected and all nodes can be visited starting from the given node.
Problem link

 

Video Tutorial

You can find the detailed video tutorial here

Thought Process

It's a basic graph problem that can be solved in 3 different ways: BFS using queue, DFS using recursion, DFS using stack(very similar to BFT using queue). The trick is using a map to keep a one-to-one mapping between the old nodes and the copied nodes, meanwhile, we can also use the map to avoid a cycle when performing BFS or DFS.

Solutions

BFS using queue

public Node cloneGraphBFS(Node node) {
if (node == null) {
return node;
}
Map<Node, Node> lookup = new HashMap<>();
Queue<Node> q = new LinkedList<>();
q.add(node);
lookup.put(node, new Node(node.val, new ArrayList<>()));
while (!q.isEmpty()) {
Node cur = q.poll();
for (Node n : cur.neighbors) {
Node copiedNeighbor = null;
if (!lookup.containsKey(n)) {
// the neighbors would be populated later when it's popped out from the queue
copiedNeighbor = new Node(n.val, new ArrayList<>());
lookup.put(n, copiedNeighbor);
q.add(n);
} else {
copiedNeighbor = lookup.get(n);
}
lookup.get(cur).neighbors.add(copiedNeighbor);
}
}
return lookup.get(node);
}
Time Complexity: O(N)
Space Complexity: O(N) the extra queue and map


DFS using recursion

public Node cloneGraphDFSWithRecursion(Node node) {
if (node == null) {
return node;
}
Map<Node, Node> lookup = new HashMap<>();
return this.cloneGraphHelper(node, lookup);
}
/**
* A dfs helper using recursion to make a copy of each node recursively via neighbors
* @param node
* @param lookup
* @return the copied node of Node
*/
private Node cloneGraphHelper(Node node, Map<Node, Node> lookup) {
if (node == null) {
return node;
}
if (lookup.containsKey(node)) {
return lookup.get(node);
}
Node copiedNode = new Node(node.val, new ArrayList<>());
lookup.put(node, copiedNode);
for (Node n : node.neighbors) {
Node copiedN = this.cloneGraphHelper(n, lookup);
copiedNode.neighbors.add(copiedN);
}
return copiedNode;
}
Time Complexity: O(N)
Space Complexity: O(N) the extra map


DFS using stack

// exactly the same as BFS except used a stack to mimic the recursion, technically any BFS can be written
// in DFS by using a stack
public Node cloneGraphDFSUsingStack(Node node) {
if (node == null) {
return node;
}
Map<Node, Node> lookup = new HashMap<>();
Node t = new Node(node.val, new ArrayList<>());
lookup.put(node, t);
Stack<Node> stack = new Stack<>();
stack.push(node);
while (!stack.isEmpty()) {
Node cur = stack.pop();
for (Node n : cur.neighbors) {
Node copiedNeighbor = null;
if (!lookup.containsKey(n)) {
copiedNeighbor = new Node(n.val, new ArrayList<>());
stack.push(n);
lookup.put(n, copiedNeighbor);
} else {
copiedNeighbor = lookup.get(n);
}
lookup.get(cur).neighbors.add(copiedNeighbor);
}
}
return lookup.get(node);
}
Time Complexity: O(N)
Space Complexity: O(N) the extra queue and stack

The main test method
public static void main(String[] args) {
CloneGraph cg = new CloneGraph();
// construct the example graph
Node n1 = new Node(1, new ArrayList<>());
Node n2 = new Node(2, new ArrayList<>());
Node n3 = new Node(3, new ArrayList<>());
Node n4 = new Node(4, new ArrayList<>());
n1.neighbors.addAll(Arrays.asList(new Node[]{n2, n4}));
n2.neighbors.addAll(Arrays.asList(new Node[]{n3, n3}));
n3.neighbors.addAll(Arrays.asList(new Node[]{n2, n4}));
n4.neighbors.addAll(Arrays.asList(new Node[]{n1, n3}));
// add a self cycle
n1.neighbors.addAll(Arrays.asList(new Node[]{n1}));
// expect the same BFS order print out
Utilities.printGraphBFS(n1);
Node copiedN1 = cg.cloneGraphBFS(n1);
Utilities.printGraphBFS(copiedN1);
}
/**
* print out a graph in BFS node, note it could be a cycle, so it's not really the topological order
*
* 1 ----- 2
* | |
* 4 ----- 3
* @param node
*/
public static void printGraphBFS(CloneGraph.Node node) {
if (node == null) {
return;
}
Set<CloneGraph.Node> visited = new HashSet<>();
Queue<CloneGraph.Node> queue = new LinkedList<>();
queue.add(node);
System.out.println(node.val);
visited.add(node);
while (!queue.isEmpty()) {
CloneGraph.Node cur = queue.poll();
for (CloneGraph.Node n : cur.neighbors) {
if (!visited.contains(n)) {
queue.add(n);
System.out.println(n.val);
visited.add(n);
}
}
}
}

References

Saturday, November 2, 2019

Leetcode solution 135: Candy

Problem Statement 

There are N children standing in a line. Each child is assigned a rating value.
You are giving candies to these children subjected to the following requirements:
  • Each child must have at least one candy.
  • Children with a higher rating get more candies than their neighbors.
What is the minimum candies you must give?
Example 1:
Input: [1,0,2]
Output: 5
Explanation: You can allocate to the first, second and third child with 2, 1, 2 candies respectively.
Example 2:
Input: [1,2,2]
Output: 4
Explanation: You can allocate to the first, second and third child with 1, 2, 1 candies respectively.
             The third child gets 1 candy because it satisfies the above two conditions.
Problem link

 

Video Tutorial

You can find the detailed video tutorial here

Thought Process

The straight forward idea is go through the ranking array, for each ranking, we try to make sure it has more points than its left and right neighbor. Note that it’s not going to work if you just do that because while you are increasing the values, it’s a chain effect that will affect the previous values as well, therefore we have to go all the way to the beginning.

Use below ranking as an example, assume everyone has 1 point already
Ranking: [5, 3, 1]

Points [1, 0, 0]
When at 5, [1, 0, 0]
When at 3, [2, 1, 0]
When at 1, [2, 2, 1]
you might think 5 points is enough, but it’s wrong. The culprit is when you at 1, and increase child at 3, you have the potential to break the assumption that child with 3 ranking always has less than its left neighbor if that neighbor has more points. Therefore, we have to go all the way to the beginning and comparison, thus resulting O(N^2) time complexity

That said, we have to go all the way to the beginning when we increase any value, so in the end it looks like [3, 2, 1]

Hint: if you cannot solve it in one pass, can we solve it in two passes?

A more clever thought is trying to distribute the candies in two passes. Previously we are trying to ensure at certain point both the left and right constrains should met. With two passes, namely first pass (from left to right) ensure all the left neighbors constrains are satisfied, and second pass ensures all the right neighbors contains are satisfied.
  • Left to right pass: Ensure I have more points than my left neighbor if my ranking is higher. Else I just have 1 candy (this is minimum)
  • Right to left pass: Ensure I have more points than my right neighbor if my ranking is higher.

Solutions

Naive solution 

// O(N^2), exceeds time limit
public int candyON2(int[] ratings) {
if (ratings == null || ratings.length == 0) return 0;
if (ratings.length == 1) return 1;
int[] candies = new int[ratings.length];
candies[0] = 1;
for (int i = 1; i < ratings.length; i++) {
// if current rating is small, then we have to go back and increase previous candies if rating is larger
if (ratings[i] < ratings[i - 1]) {
int j = i - 1;
candies[i] = 1;
// This causes a chain of effect to go back to the beginning, the simple idea only increase the neighbor
// won't work, e.g, 5, 3, 1 should be 6 instead of 5
while (j >= 0 && ratings[j] > ratings[j + 1]) {
candies[j] += 1;
j--;
}
} else if (ratings[i] > ratings[i - 1]) {
candies[i] = candies[i - 1] + 1;
} else {
candies[i] = 1;
}
}
int count = 0;
for (int i = 0; i < candies.length; i++) {
count += candies[i];
}
return count;
}
view raw CandyNaive.java hosted with ❤ by GitHub
Time Complexity: O(N^2)
Space Complexity: O(N) the extra candies array

Two pass linear solution

public int candy(int[] ratings) {
if (ratings == null || ratings.length == 0) return 0;
int n = ratings.length;
if (n == 1) return 1;
int[] candies = new int[n];
candies[0] = 1;
// left -> right, if increasing, + 1, else just assign to 1, only change the values going forward
// not changing the values we already visited(left)
for (int i = 1; i < n; i++) {
if (ratings[i - 1] < ratings[i]) {
candies[i] = candies[i - 1] + 1;
} else {
candies[i] = 1;
}
}
// right -> left, if left is larger and candy is less, +1, not changing the right part, only changing the left
for (int i = n - 1; i >= 1; i--) {
if (ratings[i - 1] > ratings[i] && candies[i - 1] <= candies[i]) {
candies[i - 1] = candies[i] + 1;
}
}
int count = 0;
for (int i = 0; i < n; i++) {
count += candies[i];
}
return count;
}
view raw Candy.java hosted with ❤ by GitHub
Time Complexity: O(N)
Space Complexity: O(N) the extra candies array

References

Saturday, October 26, 2019

Leetcode solution 201: Bitwise AND of Numbers Range

Problem Statement 

Given a range [m, n] where 0 <= m <= n <= 2147483647, return the bitwise AND of all numbers in this range, inclusive.
Example 1:
Input: [5,7]
Output: 4
Example 2:
Input: [0,1]
Output: 0

Problem link

 

Video Tutorial

You can find the detailed video tutorial here

Thought Process

Normally we just implement as we do at work, which is keep increasing the number and do an AND. It will still pass the OJ, just need to pay attention to overflow situation (using a long would solve the problem)

Now we are pushing ourselves, can we solve it more than linear. Only log/binary is faster than linear. The idea is to find the common left bits of m and n, and later shift n (total number of digits - common length digits) because the right part would end up to be 0.

For example, from 4 to 7, thte common left part is 1, the range and value would be 100 (which is n left shift twice)
  • 1 00
  • 1 01
  • 1 10
  • 1 11

Solutions

Linear solution


public int rangeBitwiseAnd(int m, int n) {
// avoid overflow
long res = m;
for (long i = (long)m + 1; i <= (long)n; i++) {
res = res & i;
if (res == 0) return 0;
}
return (int)res;
}
Time Complexity: O(N) essentially n - m
Space Complexity: O(1) no extra space is needed

Logarithmic solution


public int rangeBitwiseAnd(int m, int n) {
if (n == m) {
return n;
}
int digit = 0;
while (m != n) {
m >>= 1;
n >>= 1;
digit++;
}
return m << digit;
}
Time Complexity: O(lgN) because we keep dividing 2 (left shift) of n
Space Complexity: O(1) no extra space is needed

References

Saturday, October 12, 2019

Leetcode solution 103: Binary Tree Zigzag Level Order Traversal

Problem Statement 

Given a binary tree, return the zigzag level order traversal of its nodes' values. (ie, from left to right, then right to left for the next level and alternate between).
For example:
Given binary tree [3,9,20,null,null,15,7],
    3
   / \
  9  20
    /  \
   15   7
return its zigzag level order traversal as:
[
  [3],
  [20,9],
  [15,7]
]

Problem link

 

Video Tutorial

You can find the detailed video tutorial here

Thought Process

It’s purely an implementation problem to be honest. Very similar to to binary tree level order traversal, we just need to use some data structure to hold the values while we control the traversal order. I choose to use a deque (i.e., doubled linked list) to keep the order. You can also use a queue like many other online solutions but note when calling add(0, value) to an array list is not very efficient since to have array copy every time.

Solutions


// use deque
public List<List<Integer>> zigzagLevelOrder(TreeNode root) {
List<List<Integer>> res = new ArrayList<>();
if (root == null) {
return res;
}
Deque<TreeNode> deque = new LinkedList<>();
deque.add(root);
boolean isFromLeftToRight = true;
while (!deque.isEmpty()) {
int size = deque.size();
List<Integer> temp = new ArrayList<>();
for (int i = 0; i < size; i++) {
if (isFromLeftToRight) {
TreeNode node = deque.pollFirst();
temp.add(node.val);
if (node.left != null) {
deque.addLast(node.left);
}
if (node.right != null) {
deque.addLast(node.right);
}
} else {
TreeNode node = deque.pollLast();
temp.add(node.val);
if (node.right != null) {
deque.addFirst(node.right);
}
if (node.left != null) {
deque.addFirst(node.left);
}
}
}
res.add(temp);
isFromLeftToRight = !isFromLeftToRight;
}
return res;
}
Time Complexity: O(N) since we visit each node once
Space Complexity: O(N) since used a deque

References

Saturday, October 5, 2019

Leetcode solution 199: Binary Tree Right Side View

Problem Statement 


Given a binary tree, imagine yourself standing on the right side of it, return the values of the nodes you can see ordered from top to bottom.
Example:
Input: [1,2,3,null,5,null,4]
Output: [1, 3, 4]
Explanation:

   1            <---
 /   \
2     3         <---
 \     \
  5     4       <---

Problem link

 

Video Tutorial

You can find the detailed video tutorial here

Thought Process

This is a great problem and I highly recommend we solve it in both DFS and BFS for level order traversal because it  covers everything we need to know about trees in interviews.

This would be a better test case because right subtree would not be deep enough to cover the left subtree
   1            <---
 /   \
2     3         <---
 \     
  5             <---
[1, 3, 5] 

Level order traversal using BFS(queue)

It should be relatively natural for us think about using level order traversal and we can simply use a queue and always find the right most element when we pop out element from each level.

Level order traversal using DFS(map)

A pre-order traversal way, just like order level traversal using DFS with a map(key is depth, value is node). This method utilizes a depth to value map to record the right most value on each order while performing a pre-order traversal, namely record the node value, go left then go right. This way, right value can always overwrite the left value thus keeping a “right side view”



I personally don't like the leetcode official solution, where in DFS you don’t need two stacks and in
BFS you don’t need the maps.

Follow up, how about implement a left side view? Trivial right?

Solutions

Level order traversal using BFS(queue)


// BFS with a queue
public List<Integer> rightSideView(TreeNode root) {
List<Integer> res = new ArrayList<>();
if (root == null) {
return res;
}
// need to put node in the queue, not value
Queue<TreeNode> q = new LinkedList<>();
q.add(root);
while (!q.isEmpty()) {
int s = q.size();
// classic template for bfs, remember the queue size is the current level
for (int i = 0; i < s; i++) {
TreeNode t = q.poll();
// the last element is right side view
if (i == s - 1) {
res.add(t.val);
}
if (t.left != null) q.offer(t.left);
if (t.right != null) q.offer(t.right);
}
}
return res;
}
Time Complexity: O(N) since we visit each node once
Space Complexity: O(N), more precisely the number of element on the last level, aka queue size when it’s a complete tree


Level order traversal using DFS(map)

private void dfsHelper(Map<Integer, Integer> depthToValue, TreeNode node, int depth) {
if (node == null) {
return;
}
// this is a pre-order traversal, essentially keep overwriting the depthToValue map (right view)
// while traverse the tree from left to right
depthToValue.put(depth, node.val);
dfsHelper(depthToValue, node.left, depth + 1);
// overwrite the right side view since right side recursion comes later
dfsHelper(depthToValue, node.right, depth + 1);
}
public List<Integer> rightSideView(TreeNode root) {
Map<Integer, Integer> depthToValue = new HashMap<>();
dfsHelper(depthToValue, root, 1);
int depth = 1;
List<Integer> result = new ArrayList<>();
while (depthToValue.containsKey(depth)) {
result.add(depthToValue.get(depth));
depth++;
}
return result;
}
Time Complexity: O(N) since we visit each node once

Space Complexity: O(lgN) because we are using a map and map size should be tree height, which worst case could be O(N)

References

Saturday, September 28, 2019

Leetcode solution 1176: Diet Plan Performance

Problem Statement 

A dieter consumes calories[i] calories on the i-th day. 
Given an integer k, for every consecutive sequence of k days (calories[i], calories[i+1], ..., calories[i+k-1] for all 0 <= i <= n-k), they look at T, the total calories consumed during that sequence of k days (calories[i] + calories[i+1] + ... + calories[i+k-1]):
  • If T < lower, they performed poorly on their diet and lose 1 point; 
  • If T > upper, they performed well on their diet and gain 1 point;
  • Otherwise, they performed normally and there is no change in points.
Initially, the dieter has zero points. Return the total number of points the dieter has after dieting for calories.length days.
Note that the total points can be negative.

Example 1:
Input: calories = [1,2,3,4,5], k = 1, lower = 3, upper = 3
Output: 0
Explanation: Since k = 1, we consider each element of the array separately and compare it to lower and upper.
calories[0] and calories[1] are less than lower so 2 points are lost.
calories[3] and calories[4] are greater than upper so 2 points are gained.
Example 2:
Input: calories = [3,2], k = 2, lower = 0, upper = 1
Output: 1
Explanation: Since k = 2, we consider subarrays of length 2.
calories[0] + calories[1] > upper so 1 point is gained.
Example 3:
Input: calories = [6,5,0,0], k = 2, lower = 1, upper = 5
Output: 0
Explanation:
calories[0] + calories[1] > upper so 1 point is gained.
lower <= calories[1] + calories[2] <= upper so no change in points.
calories[2] + calories[3] < lower so 1 point is lost.

Constraints:
  • 1 <= k <= calories.length <= 10^5
  • 0 <= calories[i] <= 20000
  • 0 <= lower <= upper
Problem link

 

Video Tutorial

You can find the detailed video tutorial here

Thought Process

This is an easy problem but might be a bit hard to code it up correctly in one pass. We can use a sliding window to keep a rolling sum and compare it with upper and lower bound.

A few caveats in implementation:
  • We can simplify using two pointers start and end by keeping an index i (start) and i - k (end)
  • We can only have one loop instead of having two separate loops
sliding window

Solutions

Sliding window


public int dietPlanPerformance(int[] calories, int k, int lower, int upper) {
if (calories == null || calories.length == 0) {
return 0;
}
int rollingSum = 0;
int performance = 0;
for (int i = 0; i < calories.length; i++) {
// always adding element to the sliding window on the right
rollingSum += calories[i];
// initial value
if (i < k - 1) {
continue;
}
// when to pop out element on the left
if (i >= k) {
rollingSum -= calories[i - k];
}
if (rollingSum > upper) {
performance++;
}
if (rollingSum < lower) {
performance--;
}
}
return performance;
}
Time Complexity: O(N) visited the array once
Space Complexity: O(1) No extra space is needed

References

  • None