3 things to solve optimization problems. Optimal means either I have to find maximum answer or minimum answer.
Dynamic Programming
Divide the problem into a series of overlapping sub-problems. Then combine the solutions of sub-problems.
### Two features
1) Optimal Substructure : The problem is divisible. 2) Overlapping SubProblemss : The subproblems are repeated. >Can store them so that do not solve all the subproblems again and again. >Can use recurrence relation like Fibonacci.
Greedy
At the stage I am standing, choose minimum or maximum one. If I chose something, other’s way is not considered.
Divide and Conquer like Merge sort
divide the array into two parts, then divide it into two parts again and agian, then combine these
DP vs Greedy
DP will follow all the path(traverse all those sequence of decisions). So DP always give the optimal answer. However, Greedy doesn’t give the optimal answer always.
DP vs Divide and Conquer
Optimal substructure is same. But the main difference is Overlapping subproblems
- Best Time to Buy and Sell Stock II, 17/Jun/2024 Topic : Array, DP, Greedy couldn’t solve this within 30mi. My solution
Optimal problem is always about Dynamic Programming or Greedy!
Dynamic Programming Approach
i) dp[i] = ith maximum profit -> fail
ii) Think about state machine(all possible situations) -> succeed
The action we can do on ith day is either buy(if share is 1), or sell (if share is 0) or DO NOTHING.
Using share flag failed cuz we have to consider prev action is holding or not both.
so we have to use 2 case dp!
hold[] and notHold[]
but we always consider right after previous day.
don’t need to keep dp array, just need 2 variables.
// must buy a stock at 1st day.
hold = -prices[0];
// must do nothing at 1st day.
notHold = 0;
for(int price : prices) {
// buy(after notHold) or do nothing(after hold)
hold = max(notHold - price, hold);
// sell(after hold) or do nothing
notHold = max(hold + price, notHold);
}
//As a result, notHold is always bigger than hold.
return notHold;
public int maxProfit(int[] prices) {
// must buy a stock at 1st day.
int hold = -prices[0];
// must do nothing at 1st day.
int notHold = 0;
for(int price : prices) {
// buy(after notHold) or do nothing(after hold)
hold = Math.max(notHold - price, hold);
// sell(after hold) or do nothing
notHold = Math.max(hold + price, notHold);
}
//As a result, notHold is always bigger than hold.
return notHold;
}
Greedy Approach
Whenever we encounter a price(while loop prices), make a decision whether buy or sell or nothing. That is Greedy!
So we have to think about the criteria!
Assuming [7 1 5 3 6 4]
in stage 7(index 0), we can decide nothing cuz we don’t have a share and 1(index 1) is less.
in stage 1(index 1), decide buy cuz we don’t have a share and 5(index 2) is more.
in stage 5(index 2), decide sell cuz we have a share and 3(index 3) is less!
in stage 3, can decide buy cuz not holing and 3 is less than 6!
in stage 6, can decide sell cuz holding and 6 is more than 4!
in stage 4, can decide do nothing cuz not holding and it is last one!
so we have to check the state “hold or not” and “price of next day is more or less”.
But Assuming [1 2 3 4 5]
if we use above criteria. we get max profit is only 2 not 4.
so Most important Thing is that decision to sell is when only find last one of a consecutive increasing prices.
class Solution {
public int maxProfit(int[] prices) {
int maxProfit = 0;
int day = 0;
int buy = 0;
int sell = 0;
// last day cannot do anything.
while(day < prices.length - 1) {
// Do nothing until finding decreasing moment!
// If found decreasing, then Buy!
// And always sell occurs after buy cuz the codes order! so don't need to keep track of a state "hold or not".
while(day < (prices.length - 1) && prices[day] >= prices[day + 1] ) {
day++;
}
buy = day;
// Decide sell
while(day < (prices.length - 1) && prices[day] <= prices[day + 1]) {
day++;
}
sell = day;
maxProfit = maxProfit - prices[buy] + prices[sell];
}
return maxProfit;
}
}
- 45. Jump Game II, 22/Jun/2024 Topic : Array, DP, Greedy could solve this within 30mi using DP method. My solution
Intuition
Jump Game I is about true or not if from nums[0] to nums[last index].
Jump Game II is about the minimum number of jump to reach mums[last index]!
and always can reach to last!
We can come up with DP method and Greedy when we see “maximum” or “minimum” word from problem.
But we can also come up with BFS when we see “shortest path(route)” words from problem or problem implying that meaning without direct words.
Approach
-
DP for mimimum number of jump!
public int jump(int[] nums) { // dp[i] = The minimum num of jumps to reach nums[i]; int n = nums.length; int[] dp = new int[n]; Arrays.fill(dp, Integer.MAX_VALUE); dp[0] = 0; for (int i = 0; i < n; i++) { for(int jump = 0; jump <= nums[i] && (i + jump) < n; jump++) { dp[i + jump] = Math.min(dp[i + jump], dp[i] + 1); } } return dp[n-1]; }
This way can solve it cuz nums[i] < 10310^3103. so 10710^7107 don’t exceed the time limit.
==But if the nums[i] would be bigger or nums length longer?…==
Then!
- BFS for finding shortest route!
We know a template of BFS like below.
visited = []
q = deque()
q.append(root)
while q:
cur_node = q.popleft()
visited.append(cur_node.value)
if cur_node.left:
q.append(cur_node.left)
if cur_node.right:
q.append(cur_node.right)
return visited
But this problem is quite different with typical BFS problem.
we don’t need to route(visited) but focusing on the jump count in that case.
So we can swich visited array to a valuable containing minimum jump count until certain index!
And we cannot search from current node to left and right node. But we can search(go) from current index to Possible Jump Position(index).
In BFS, If we cannot reach destination in a certain level (of graph).
Then we change cur_node to next level node and repeat to travel. and we add the cur_node to visited array! ==We don’t care whether cur_node is left or right. We only focus the fact one more level search is needed!==
Like that, if we cannot reach last index(target) from current index(of array).
We have to jump from curr index to the next level index. and we update possble way(reachable indice is now current level range which will be searched!)
Then how can we know what is the next level index? it means how can we know the range current search indice range.
First, start the MaxReach = nums[0]
.
Then curr search range(curr level) is now [1, MaxReach] (with 1 jump from 0 index).
Then we have to upgrade maxReach while travel on [1, MaxReach].
But while traveling, MaxReach can be changed. so we have to hold one temp valuable for MaxReach.
if current level travel ended, MaxReach will be next level index! with increasing jump by 1!
if we can reach last index then break! return jumps!
Be careful that MaxReach index doesn’t mean best next jump index. it means just for same level(jump) range end point!
So that way like BFS, we can find shortest path(minimum num of jumps)!
Okay. Let’s write the code~
class Solution {
public int jump(int[] nums) {
// Minimum num of jumps
int jumps = 0;
// To set next level range
int maxReach = 0;
// Store maxReach temparily.
int next = 0;
int n = nums.length;
for (int i = 0; i < n; i++) {
// for check end, We have to use `next` not `maxReach` because the jump timing is the time when the `next` update.
if (next < n - 1) {
// set the search range.
maxReach = Math.max(maxReach, i + nums[i]);
// if current level(on same jump) travel ended, MaxReach will be next level index! with increasing jump by 1!
if (i == next) {
next = maxReach;
jumps++;
}
}
else break;
}
return jumps;
}
}
Intervals (but I think this is about Greedy and Array)
Oh… There is
Interval
Datatype in Java!! it means[start, end]
- 56. Merge Intervals, 30/Jun/24 Topics : Array, Sorting couldn’t solve it in 30mi, but I didn’t know How to sort and nested array and How to convert List to Array. So I searched it… 전에 백준에서 풀었던 미팅룸 최대로 잡기? 뭐 그런 문제랑 비슷함. 그때도 그리디였음. 포문 돌고 요소 마주할 때마다 최선의 선택을 하는 것, 그 조건을 생각해내는게 어렵긴해.
class Solution {
public int[][] merge(int[][] intervals) {
ArrayList<int[]> ans = new ArrayList<int[]>();
//intervals = Arrays.stream(intervals).sorted(Comparator.comparingInt(arr -> arr[0])).toArray(int[][]::new);
//Arrays.sort(intervals, (i1, i2) -> Integer.compare(i1[0], i2[0]));
Arrays.sort(intervals, (a, b) -> a[0] - b[0]);
ans.add(new int[] {intervals[0][0], intervals[0][1]});
for (int i = 1; i < intervals.length; i++) {
int len = ans.size();
// Overlapping occurs
if (ans.get(len - 1)[1] >= intervals[i][0]) {
//if (ans.get(len - 1)[1] < intervals[i][1]) ans.get(len - 1)[1] = intervals[i][1];
ans.get(len - 1)[1] = Math.max(ans.get(len - 1)[1], intervals[i][1]);
}
else ans.add(new int[] {intervals[i][0], intervals[i][1]});
}
//return ans.stream().toArray(int[][]::new);
return ans.toArray(new int[ans.size()][]);
}
}
- 57. Insert Interval, 1/Jul/24
Topics : Array
couldn’t solve it 1h… Why… I don’t know…
I just wanted to use only one loop. But using
else if
so many times is so confusing. I can’t think about all possible errors…
So I have to think very straight-forward way like this.
class Solution {
public int[][] insert(int[][] intervals, int[] newInterval) {
List<int[]> ans = new ArrayList<>();
int len = intervals.length;
int i = 0;
// Add all intervals before newInterval starts.
while (i < len && intervals[i][1] < newInterval[0]) {
ans.add(intervals[i]);
i++;
}
// Merge all overlapping intervals using newInterval
while (i < len && intervals[i][0] <= newInterval[1]) {
newInterval[0] = Math.min(intervals[i][0], newInterval[0]);
newInterval[1] = Math.max(intervals[i][1], newInterval[1]);
i++;
}
// Insert(add) the newInterval on itself or merged state
ans.add(newInterval);
// Add all the rest
while (i < len) {
ans.add(intervals[i]);
i++;
}
return ans.toArray(new int[ans.size()][]);
}
}
I tried to solve it using one for
loop like below.
public List<Interval> insert(List<Interval> intervals, Interval newInterval) {
List<Interval> result = new ArrayList<Interval>();
for (Interval i : intervals) {
if (newInterval == null || i.end < newInterval.start)
result.add(i);
else if (i.start > newInterval.end) {
result.add(newInterval);
result.add(i);
// Use newInterval as a Flag.
newInterval = null;
} else {
newInterval.start = Math.min(newInterval.start, i.start);
newInterval.end = Math.max(newInterval.end, i.end);
}
}
if (newInterval != null)
result.add(newInterval);
return result;
}
- 452. Minimum Number of Arrows to Burst Balloons, 1/Jul/24 Topics : #Array, #Greedy, #Sorting could solve it using Hint… (Keep track intersect range, and Narrow it) Actually That hint is essence of this problem.
Be careful that when you sort, i1- i2 can be overflow So use `Interger.compare(i1, i2) more
class Solution {
public int findMinArrowShots(int[][] points) {
Arrays.sort(points, (i1, i2) -> Integer.compare(i1[0], i2[0]));
int minShot = 1;
// keep track of intersect range;
int start = points[0][0];
int end = points[0][1];
for (int i = 1; i < points.length; i++) {
// shot is not included in next balloon range.
if (end < points[i][0]) {
minShot++;
start = points[i][0];
end = points[i][1];
}
// narrow possible shot range.
else {
start = Math.max(start, points[i][0]);
end = Math.min(end, points[i][1]);
}
}
return minShot;
}
}
Actually, we don’t need to keep track of start
variable bc we sort points
in increasing order of start
postion.
This is the solution based on my first idea.
I thought We have to choose points[i][1]
and keep that as possible as that arrow knock down next balloon. It is based on Greedy. Do the best choice on current index(stage).
We should shoot as to the right as possible, because since balloons are sorted, this gives you the best chance to take down more balloons.
plus) sort first points
in order of the increasing end
position.
public int findMinArrowShots(int[][] points) {
if (points.length == 0) {
return 0;
}
Arrays.sort(points, (a, b) -> Integer.compare(a[1], b[1]));
int arrowPos = points[0][1];
int arrowCnt = 1;
for (int i = 1; i < points.length; i++) {
if (arrowPos >= points[i][0]) {
continue;
}
arrowCnt++;
arrowPos = points[i][1];
}
return arrowCnt;
}
- 148. Sort List, 22/Jul/24
Topics : #LinkedList , #TwoPointers , #DividAndConquer , #Sorting , #MergeSort
couldn’t solve it within 2h. It was so difficult to me.
I forgot how to sort by Merge or Quick. So I have to learn them again.
And The main difficulty if to do merge sort using constant space $O(1)$
At the first, I used
left.val variable
notleft
itself. This problem is about LinkedList. So I should’ve used to use traits of that! And I started tomergeSort
not usingsortList
at all. That is also failure point. ^0c80f8
public ListNode sortList(ListNode head) {
// base code
if (head == null || head.next == null) return head;
// divide two sublist using Two pointers.
ListNode slow = head; // To find mid
ListNode fast = head.next.next;
while (fast != null && fast.next != null) {
slow = slow.next;
fast = fast.next.next;
}
// For dividing disconnect two sublist.
ListNode rightStart = slow.next;
slow.next = null;
ListNode left = sortList(head);
ListNode right = sortList(rightStart);
return merge(left, right);
}
private ListNode merge(ListNode left, ListNode right) {
ListNode dummy = new ListNode();
ListNode curr = dummy;
// using traits of LiknedList
while (left != null && right != null) {
if (left.val < right.val) {
curr.next = left;
left = left.next;
}
else {
curr.next = right;
right = right.next;
}
curr = curr.next;
}
// if left sublist is left to be unprocessed.
if (left != null) curr.next = left;
// if left sublist is left, just connect two
else curr.next = right;
return dummy.next;
}
- 427. Construct Quad Tree, 22/Jul/24 Topics : #Array , #DividAndConquer , #Tree , #Matrix could solve it in 34mi bc it is typical DAC and I’ve solve a lot this type problem!
i ) It is easy, just Follow the step described by problem. (my code)
Time Complexity : $O(N^2 * logN)$
If we have a grid, size 4. then we have to visit 4^2 cell to check has same val
.
size | visit cnt |
---|---|
4 | 16 |
2 | 16 |
1 | 16 |
So we have to visit every cell(N^2) logN times! And my code doesn’t use base code that is essential in recursion.
public Node construct(int[][] grid) {
return quadTree(new Node(), 0, 0, grid.length, grid);
}
private Node quadTree(Node curr, int row, int col, int size, int[][] grid) {
// Check if grid has the same value
int res = hasSame(row, col, size, grid);
if (res != -1) {
curr.isLeaf = true;
curr.val = (res == 1) ? true : false ;
}// if grid has different value then divide it into 4 sub-grids
else {
curr.isLeaf = false;
curr.val = true; // any value
curr.topLeft = quadTree(new Node(), row, col, size/2, grid);
curr.topRight = quadTree(new Node(), row, col+size/2, size/2, grid);
curr.bottomLeft = quadTree(new Node(), row+size/2, col, size/2, grid);
curr.bottomRight = quadTree(new Node(), row+size/2, col+size/2, size/2, grid);
}
return curr;
}
private int hasSame(int row, int col, int size, int[][] grid) {
int check = grid[row][col];
for (int i = row; i < row + size; i++) {
for (int j = col; j < col + size; j++) {
if (check != grid[i][j]) return -1;
}
}
return (check == 0) ? 0 : 1;
}
ii ) Best solution using base code
Time complexcity : $O(N^2)$, bc we don’t need to visit all cell every recursion! (without hasSame
function). We visit only each cell just once.
It works actually post order(meaning reverse).
It divide whole grid into by 1 sized grid.
And then check if return node is leaf
and val
is same with other 3 part.
Wow so genious.
public Node construct(int[][] grid) {
return helper(grid, 0, 0, grid.length);
}
private Node helper(int[][] grid, int x, int y, int len) {
if (len == 1) {
return new Node(grid[x][y] != 0, true, null, null, null, null);
}
Node result = new Node();
Node topLeft = helper(grid, x, y, len / 2);
Node topRight = helper(grid, x, y + len / 2, len / 2);
Node bottomLeft = helper(grid, x + len / 2, y, len / 2);
Node bottomRight = helper(grid, x + len / 2, y + len / 2, len / 2);
if (topLeft.isLeaf && topRight.isLeaf && bottomLeft.isLeaf && bottomRight.isLeaf
&& topLeft.val == topRight.val && topRight.val == bottomLeft.val && bottomLeft.val == bottomRight.val) {
result.isLeaf = true;
result.val = topLeft.val;
} else {
result.topLeft = topLeft;
result.topRight = topRight;
result.bottomLeft = bottomLeft;
result.bottomRight = bottomRight;
}
return result;
}
- 198. House Robber, 2/Aug/24 Topics : #Array , #DynamicProgramming could solve it in 19mi bc it is typical DP.
The main Idea is to start with Greedy. Choose the best(max) at that house. Should I steal this house or not? For that answer, I have to store previous best(max amount) So I decided that it is DP not greedy!
public int rob(int[] nums) {
/**
dp[i] = max amount until `i`th house
Math.max(including `i`th houst, not including `i`th house)
Math.max(dp[i-2] + nums[i], dp[i-1]);
*/
int n = nums.length;
int[] dp = new int[n];
dp[0] = nums[0];
if (n > 1) dp[1] = Math.max(nums[0], nums[1]);
if (n > 2) {
for (int i = 2; i < n; i++) {
dp[i] = Math.max(dp[i-2] + nums[i], dp[i-1]);
}
}
return dp[n-1];
}
constant space
Just switch
dp[i-2] -> prepreMax dp[i-1] -> preMax ```java public int rob(int[] nums) { int prepreMax = nums[0]; if (nums.length == 1) return nums[0]; if (nums.length == 2) return Math.max(nums[0], nums[1]); int preMax = Math.max(nums[0], nums[1]);
for (int i = 2; i < nums.length; i++) {
int temp = preMax;
preMax = Math.max(preMax, prepreMax + nums[i]);
prepreMax = temp;
}
return preMax;
} ``` more fantastic code ```java public int rob(int[] num) {
int prevNo = 0;
int prevYes = 0;
for (int n : num) {
int temp = prevNo;
prevNo = Math.max(prevNo, prevYes);
prevYes = n + temp;
}
return Math.max(prevNo, prevYes); } ```
- 139. Word Break, 2/Aug/24 Topics : #Array , #HashTable , #String , #DynamicProgramming , #Trie , #Memoization couldn’t solve it bc of the edge case… but my idea is right.
Two properties of Dynamic Programming 1) overlapping subproblem 2) divisible subproblem
if (cat, sand) is failed already, we don’t have to care about (cats, and) case. It will be failed also. So I want to store(memoization) every result to prevent redundant same process.
i) Using recursion
public boolean wordBreak(String s, List<String> wordDict) {
HashSet<String> dictSet = new HashSet<>();
for (String word : wordDict) dictSet.add(word);
// dp[i] = s.substring(0, i) can make segemenerk or not.
Boolean[] dp = new Boolean[s.length()];
Arrays.fill(dp, null);
return helper(s, 0, dp, dictSet);
}
private boolean helper(String s, int start, Boolean[] dp, HashSet<String> dictSet) {
if (start == s.length()) return true;
if (dp[start] != null) return dp[start];
for (int i = start; i < s.length(); i++) {
if (dictSet.contains(s.substring(start, i+1)) == true) {
boolean res = helper(s, i+1, dp, dictSet);
if (res == true) {
dp[start] = true;
return true;
}
}
}
dp[start] = false;
return false;
}
ii) Using iteratively
public boolean wordBreak(String s, List<String> wordDict) {
// dp[i] = s.substring(0, i) can make segemenerk or not.
boolean[] dp = new boolean[s.length() + 1];
Arrays.fill(dp, false);
// default base
dp[0] = true;
for (int i = 0; i < s.length(); i++) {
// prevent to useless case
if (!dp[i]) continue;
// already true then break!
if (dp[s.length()] == true) return true;
for(String word : wordDict) {
int end = i + word.length();
if (end <= s.length() && s.substring(i, end).equals(word)) {
dp[end] = dp[i];
}
}
}
return dp[s.length()];
}
- 322. Coin Change, 3/Aug/24
Topics : #Array , #DynamicProgramming , #BFS
couldn’t solve it in 1h…
I solved this before, but forgot…
[[2023-12-10-day44#^6c6981 | Same problem]]
dp[i] = fewest number of coins to make up
i
amount money.
The main idea is for make up
i
amount money, we have to choose a coin at the last. So we can express that dp[i] = 1 + dp[i - coin] ```java public int coinChange(int[] coins, int amount) { int[] dp = new int[amount + 1]; Arrays.fill(dp, -1); dp[0] = 0; // the cast in which Choose nothing.
for (int i = 1; i <= amount; i++) {
for (int coin : coins) {
if (coin <= i && dp[i - coin] != - 1) {
if (dp[i] == - 1) dp[i] = 1 + dp[i - coin];
else dp[i] = Math.min(dp[i], 1 + dp[i - coin]);
}
}
}
return dp[amount];
} ```
ii) Using BFS
Fewest(minimum) coins can be interpreted that find shortest path.
public int coinChange(int[] coins, int amount) {
ArrayDeque<Integer> q = new ArrayDeque<>();
int minCoin = 0;
boolean[] visited = new boolean[amount+1];
q.add(0);
visited[0] = true;
while (!q.isEmpty()) {
int size = q.size();
while (size-- > 0) {
int total = q.pop();
if (total == amount) return minCoin;
for (int coin : coins) {
if (coin <= amount && total + coin <= amount && visited[total+coin] == false) {
q.add(total + coin);
visited[total+coin] = true;
}
}
}
minCoin++;
}
return -1;
}
- 300. Longest Increasing Subsequence, 4/Aug/24
Topics : #Array , #DynamicProgramming , #BinarySearch
could solve it in 38mi using hint.
dp[i] : LIS including
i
index.
I know I already solved it… but also forgot ha….
public int lengthOfLIS(int[] nums) {
int[] dp = new int[nums.length];
dp[0] = 1;
int longest = 1;
for (int i = 1; i < nums.length; i++) {
dp[i] = 1;
for (int j = 0; j < i; j++) {
if (nums[i] > nums[j]) dp[i] = Math.max(dp[i], dp[j] + 1);
}
if (longest < dp[i]) longest = dp[i];
}
return longest;
}
This code TC is $O(n^2)$. So we have to reduce it until $O(nlogn)$.
ii) Using binary search
public int lengthOfLIS(int[] nums) {
ArrayList<Integer> ans = new ArrayList<>();
ans.add(nums[0]);
for (int num : nums) {
int last = ans.get(ans.size() - 1);
if (last < num) ans.add(num);
else {
// find `num` position in ans.
replace(ans, num);
}
}
return ans.size();
}
private void replace(ArrayList<Integer> ans, int num) {
// start binarysearch and replace.
int start = 0, end = ans.size() - 1;
while (start < end) {
int mid = (start + end) / 2;
if (ans.get(mid) == num) return;
else if (ans.get(mid) < num) start = mid + 1;
else end = mid;
}
ans.set(start, num);
}
else if (ans.get(mid) < num) start = mid + 1;
else end = mid;
This part is important!
imagine ans =
[3, 4, 6]
and you want put5
to ans. the right position is 2(value 6). ifans[mid] > num
then def mid can’t be the right position. sostart = mid + 1
. But ifans[mid] < num
, then mid can be the right position. soend = mid
Using size
, don’t need to use ArrayList
.
public int lengthOfLIS(int[] nums) {
int[] tails = new int[nums.length];
int size = 0;
for (int x : nums) {
int i = 0, j = size;
while (i != j) {
int m = (i + j) / 2;
if (tails[m] < x)
i = m + 1;
else
j = m;
}
tails[i] = x;
if (i == size) ++size;
}
return size;
}
- 120. Triangle, 5/Aug/24
Topics : #Array , #DynamicProgramming
I know, I got this problem before.
could solve it in 45mi bc it was hard to handle the
ArrayList array
.public int minimumTotal(List<List<Integer>> triangle) { // dp[i][j] = min path sum until `j` index on `i` row and including `j` index. int n = triangle.size(); ArrayList<Integer>[] dp = new ArrayList[n]; dp[0] = new ArrayList<Integer>(); dp[0].add(triangle.get(0).get(0)); for (int i = 1; i < n; i++) { dp[i] = new ArrayList<>(); for (int j = 0; j < triangle.get(i).size(); j++) { int curr = triangle.get(i).get(j); if (j == 0) dp[i].add(dp[i-1].get(j) + curr); else if (j == triangle.get(i).size() - 1) dp[i].add(dp[i-1].get(j-1) + curr); else dp[i].add(Math.min(dp[i-1].get(j-1) + curr, dp[i-1].get(j) + curr)); } } int min = Integer.MAX_VALUE; for (int i = 0; i < dp[n - 1].size(); i++) { min = Math.min(min, dp[n - 1].get(i)); } return min; }
This code has $O(row * col)$ space. But follow up question ask using only $O(row)$.
Bottom up method
top-down method can’t solve it on int[n]
because when we overwrite dp[j]
we can’t use that for dp[j+1]
. Even if using temp
we can’t use dp[j+1]
to next loop
. So we have to use bottom up
bc dp[i][j]
is only used for dp[i-1][j]
. So we can overwrite it!
public int minimumTotal(List<List<Integer>> triangle) {
// dp[i] : min path sum until `i` index on row and including `i` index.
int n = triangle.size();
int[] dp = new int[n];
for (int i = 0; i < n; i++) dp[i] = triangle.get(n-1).get(i);
// bottom-up
for (int i = n - 2; i >= 0; i--) {
for (int j = 0; j <= i; j++) {
dp[j] = Math.min(dp[j], dp[j+1]) + triangle.get(i).get(j);
}
}
return dp[0];
}
So it can run on $O(1)$ space if using triangle
itself.
- 64. Minimum Path Sum, 5/Aug/24 Topics : #Array , #DynamicProgramming , #Matrix
public int minPathSum(int[][] grid) {
int m = grid.length;
int n = grid[0].length;
int[][] dp = new int[m][n];
dp[0][0] = grid[0][0];
// set all topmost and leftmost cell.
for (int col = 1; col < n; col++) dp[0][col] = dp[0][col-1] + grid[0][col];
for (int row = 1; row < m; row++) dp[row][0] = dp[row-1][0] + grid[row][0];
for (int row = 1; row < m; row++) {
for (int col = 1; col < n; col++) {
dp[row][col] = Math.min(dp[row-1][col], dp[row][col-1]) + grid[row][col];
}
}
return dp[m-1][n-1];
}
We can reduce space complexity to $O(1)$ overwriting graph
itself .
- 63. Unique Paths II, 5/Aug/24
Topics : #Array , #DynamicProgramming , #Matrix
could solve it in 25mi.
The important thing is when setting the
row and col = 0
, we mustif (obstacleGrid[row][0] == 1) break;
to preventdp[row][0] = 1
when there is a obstacle before that row.
public int uniquePathsWithObstacles(int[][] obstacleGrid) {
int m = obstacleGrid.length;
int n = obstacleGrid[0].length;
int[][] dp = new int[m][n];
// Set [row][0]
for (int row = 0; row < m; row++) {
Arrays.fill(dp[row], 0);
if (obstacleGrid[row][0] == 1) break;
dp[row][0] = 1;
}
// Set [0][col]
for (int col = 0; col < n; col++) {
if (obstacleGrid[0][col] == 1) break;
dp[0][col] = 1;
}
for (int i = 1; i < m; i++) {
for (int j = 1; j < n; j++) {
if (obstacleGrid[i][j] == 1) continue;
dp[i][j] = dp[i-1][j] + dp[i][j-1];
}
}
return dp[m-1][n-1];
}
using obstacleGrid
itself
public int uniquePathsWithObstacles(int[][] obstacleGrid) {
int m = obstacleGrid.length;
int n = obstacleGrid[0].length;
if (obstacleGrid[0][0] == 1) obstacleGrid[0][0] = 0;
else obstacleGrid[0][0] = 1;
for (int col = 1; col < n; col++) {
if (obstacleGrid[0][col] == 1) obstacleGrid[0][col] = 0;
else obstacleGrid[0][col] = obstacleGrid[0][col-1];
}
for (int row = 1; row < m; row++) {
if (obstacleGrid[row][0] == 1) obstacleGrid[row][0] = 0;
else obstacleGrid[row][0] = obstacleGrid[row-1][0];
}
for (int i = 1; i < m; i++) {
for (int j = 1; j < n; j++) {
if (obstacleGrid[i][j] == 1) obstacleGrid[i][j] = 0;
else obstacleGrid[i][j] = obstacleGrid[i-1][j] + obstacleGrid[i][j-1];
}
}
return obstacleGrid[m-1][n-1];
}
Using 1D dp
wow so difficult to understand. I wonder what happen to row[0]. if 0
index of previous row has obstacle, then row[0] = 0
, after that row[0]
is always 0
!! So brilliant! We don’t touch row[0]
except for that cell has obstacle.
class Solution {
public int uniquePathsWithObstacles(int[][] obstacleGrid) {
int n = obstacleGrid[0].length;
int[] dp = new int[n];
dp[0] = 1;
for (int[] row : obstacleGrid) {
for (int i = 0; i < n; i++) {
if (row[i] == 1) dp[i] = 0;
else {
if (i > 0) dp[i] = dp[i] + dp[i-1];
}
}
}
return dp[n-1];
}
}
- 5. Longest Palindromic Substring, 6/Aug/24 Topics : #TwoPointers , #String , #DynamicProgramming
The key point here is that Think about 2 cases, Odd length and Even length!
i) Using DP
TC : $O(n^2)$ SC : $O(n^2)$
The important thing is that using
dp[l+1][r-1]
. We iterater
from0
ton-1
, so when we checkdp[l][r]
, we can usedp[0~(r-1)][r-1]
bcr-1
is previous step, already calculated(checked)!! ```java public String longestPalindrome(String s) { int n = s.length(); // dp[i][j] : index fromi
toj
on String s has pelindrome or not boolean[][] dp = new boolean[n][n]; String ans = “”;
for (int r = 0; r < n; r++) {
// alone word is pelindrome.
if (ans.length() < 1) ans = s.substring(r,r+1);
dp[r][r] = true;
for (int l = 0; l < r; l++) {
// Odd length case
if (s.charAt(l) == s.charAt(r) && (r-l == 2 || dp[l+1][r-1])) {
dp[l][r] = true;
if (r-l+1 > ans.length()) ans = s.substring(l, r+1);
}
// Even length case
if (s.charAt(l) == s.charAt(r) && (r-l == 1 || dp[l+1][r-1])) {
dp[l][r] = true;
if (r-l+1 > ans.length()) ans = s.substring(l, r+1);
}
}
}
return ans;
} ```
ii) Using intuitive
So beautiful code. TC : $O(n^2)$ SC : $O(1)$ <—- BEST!!
public String longestPalindrome(String s) {
String ans = "";
for (int center = 0; center < s.length(); center++) {
String odd = expand(s, center, center);
String even = expand(s, center, center + 1);
if (odd.length() > ans.length()) ans = odd;
if (even.length() > ans.length()) ans = even;
}
return ans;
}
private String expand(String s, int left, int right) {
while (left >= 0 && right < s.length() && s.charAt(left) == s.charAt(right)) {
left--;
right++;
}
// return "restored pelindrom"
return s.substring(left + 1, right);
}
- 97. Interleaving String, 7/Aug/24 Topics : #String , #DynamicProgramming couldn’t solve it in 1h.
The main idea is
dp[i][j] = s3[:i+j] can be formed by interleaving s1[:i] and s2[:j] or not
Using 2D DP
public boolean isInterleave(String s1, String s2, String s3) {
//if (s1.length() == 0 && s2.length() == 0) return (s3.length() == 0);
if (s1.length() + s2.length() != s3.length()) return false;
// dp[i][j] = s3[:i+j] can be formed by interleaving s1[:i] and s2[:j]
boolean[][] dp = new boolean[s1.length() + 1][s2.length() + 1];
dp[0][0] = true;
for (int i = 1; i <= s1.length(); i++) dp[i][0] = (dp[i-1][0] && s1.charAt(i-1) == s3.charAt(i-1));
for (int j = 1; j <= s2.length(); j++) dp[0][j] = (dp[0][j-1] && s2.charAt(j-1) == s3.charAt(j-1));
for (int i = 1; i <= s1.length(); i++) {
for (int j = 1; j <= s2.length(); j++) {
dp[i][j] = (dp[i-1][j] && s3.charAt(i+j-1) == s1.charAt(i-1))
|| (dp[i][j-1] && s3.charAt(i+j-1) == s2.charAt(j-1));
}
}
return dp[s1.length()][s2.length()];
}