Skip to content

Latest commit

 

History

History
857 lines (615 loc) · 41.3 KB

README.md

File metadata and controls

857 lines (615 loc) · 41.3 KB

Leetcode-Practice

Join me on my leetcode journey. Problems are based off the Neetcode 150. Will look to do more after interviews. Read more here: (https://neetcode.io/practice)

Not only is LeetCode a great way to prepare for interviews, but it's a wonderful tool to learn a new language. As a result, to develop my proficency with Python, I will be completing tasks mostly in Python via LeetCode and CodeWars as well.

  • Side note: I will be on vacation from May 16th-May 31. Can't gurantee that I will be able to commit every day but I am definitley going to do a lot to prepare. Additionally, I will be starting my internship on June 10th!
  • Additionally, the WPI Investing Association will atart their summer project on June 1 so I'll do my best to stay consistent on this repo.

  • Progress report 5/29/2024: Honestly, some of the easy problems are starting to actually feel easy. Trying to do the Mediums is tricky, and definitley intimidating at first.

Notes for specific problems:

EASY: [Problem 1]: Invert Binary Tree: Given the root of a binary tree, invert the tree, and return its root.

*Notes on Problem

  • My implementation is a Post Order Traversal of the Tree
  • The implementation uses recursion to find the leaf nodes, which then return the parent node to allow swapping
  • Set the value of the left and right nodes to new TreeNode variables
  • We switch the left and the right of the roots to ivnert the tree properly

image


EASY: [Problem 2]: Binary Search: Given an array of integers nums which is sorted in ascending order, and an integer target, write a function to search target in nums. If target exists, then return its index. Otherwise, return -1.

You must write an algorithm with O(log n) runtime complexity.

Example 1:

Input: nums = [-1,0,3,5,9,12], target = 9 Output: 4 Explanation: 9 exists in nums and its index is 4

*Notes on Problem

  • Think of the search like an array where you split the size each time you finish iterating the while loop
  • Look closely at how we determine the middle of the array. It is low + ((high -low)/2) You initially thought that low was included in the parenthesis
  • We set val to the middle value to determine if our target is higher or lower than our middle value
  • If the target is higher, then we need to increase the lower bound of the array by middle+1
  • If the target is lower than the middle, we need to decrease the upper bound of the array by middle-1
  • If it's neither, we are at the target! Return the index.

EASY: [Problem 3]: Two sum: Given an array of integers nums and an integer target, return indices of the two numbers such that they add up to target.

You may assume that each input would have exactly one solution, and you may not use the same element twice.

You can return the answer in any order.

Example 1:

Input: nums = [2,7,11,15], target = 9 Output: [0,1] Explanation: Because nums[0] + nums[1] == 9, we return [0, 1].

*Notes on Problem *This solution is O(n^2). When you redo, can you think of a better solution using hashmap? Maybe store target in a hashmap? *Double for loop that returns the indexes the two numbers in a array that add up to target

  • i!=j ensures we are not adding up the same element

*Notes ON REVAMPED O(N) SOLUTION

  • Optimized solution this time, O(n) instead of O(n^2)
  • create a hashmap
  • loop through array, constantly updating a variable equal to "target - nums[i]" after every iteration
  • if the variable created is present in the hashmap
  • set the start index to the map.get(variable)
    
  • set the second index to the current index
    
  •      break to reduce runtime
    
  • Outside of the if, add the current value and its index to map
  • (added at the end to avoid self reference)

EASY: [Problem 4]: firstBadVersion: You are a product manager and currently leading a team to develop a new product. Unfortunately, the latest version of your product fails the quality check. Since each version is developed based on the previous version, all the versions after a bad version are also bad.

Suppose you have n versions [1, 2, ..., n] and you want to find out the first bad one, which causes all the following ones to be bad.

You are given an API bool isBadVersion(version) which returns whether version is bad. Implement a function to find the first bad version. You should minimize the number of calls to the API.

Example 1:

Input: n = 5, bad = 4 Output: 4 Explanation: call isBadVersion(3) -> false call isBadVersion(5) -> true call isBadVersion(4) -> true Then 4 is the first bad version.

*Notes

  • This is a binary search problem as we want to minimize function calls, so instead of O(n) with binary search it is O( log n)
    • Everything about the binary search stays the same. However, to minimize isBadVersion calls, if we're at a bad version in the middle, we make our upper bound the middle.
    • our answer will reside in our variable low. This is because we keep decreasing high if it's a bad version, and increasing low if not. At the point where low>high, we found the bad version

EASY: [Problem 5]: Valid Parentheses: Given a string s containing just the characters '(', ')', '{', '}', '[' and ']', determine if the input string is valid.

An input string is valid if:

Open brackets must be closed by the same type of brackets. Open brackets must be closed in the correct order. Every close bracket has a corresponding open bracket of the same type.

Example 1:

Input: s = "()" Output: true Example 2:

Input: s = "()[]{}" Output: true Example 3:

Input: s = "(]" Output: false

*Notes

  • This is a notroious stack problem. How do we recognize this?
  • We can notice this by seeing that the answer directly depends on a corresponding element
  • Each time we have a opening parentheses, push the closing parenthesis to the stack
  • If the stack is empty or we don't find the closing parenthesis next, we return false
  • Else, it's true.

EASY: [Problem 6]: Best Stock Price: You are given an array prices where prices[i] is the price of a given stock on the ith day.

You want to maximize your profit by choosing a single day to buy one stock and choosing a different day in the future to sell that stock.

Return the maximum profit you can achieve from this transaction. If you cannot achieve any profit, return 0.

Example 1:

Input: prices = [7,1,5,3,6,4] Output: 5 Explanation: Buy on day 2 (price = 1) and sell on day 5 (price = 6), profit = 6-1 = 5. Note that buying on day 2 and selling on day 1 is not allowed because you must buy before you sell. Example 2:

Input: prices = [7,6,4,3,1] Output: 0 Explanation: In this case, no transactions are done and the max profit = 0.

*Notes

  • Set the smallest to the first element
  • Loop through, if current stock is less than smallest set current to smallest
  • If the next element minus smalelst is greater than a max variable
  • Set max variable to next element minus smallest
  • Return max variable

EASY: [Problem 7]: Valid Palindrome: A phrase is a palindrome if, after converting all uppercase letters into lowercase letters and removing all non-alphanumeric characters, it reads the same forward and backward. Alphanumeric characters include letters and numbers.

Given a string s, return true if it is a palindrome, or false otherwise.

Example 1:

Input: s = "A man, a plan, a canal: Panama" Output: true Explanation: "amanaplanacanalpanama" is a palindrome. Example 2:

Input: s = "race a car" Output: false Explanation: "raceacar" is not a palindrome.

*Notes

  • s.replaceAll("\p{Punct}", ") will remove punctuations, and .toLowerCase and replaceAll (" ", "") will remove all spaces and convert uppercase to lowercase
  • This could honestly be a stack problem, maybe redo as a stack question
  • Create a new string and add letters from back to front
  • Look very closely for syntax to adding from back to front
  • If that string is equal to original, true, if not, false.

EASY: [FIRST DYNAMIC PROGRAMMING PROBLEM: Problem 8]: Climbing Stairs: You are climbing a staircase. It takes n steps to reach the top.

Each time you can either climb 1 or 2 steps. In how many distinct ways can you climb to the top?

Example 1:

Input: n = 2 Output: 2 Explanation: There are two ways to climb to the top.

  1. 1 step + 1 step
  2. 2 steps Example 2:

Input: n = 3 Output: 3 Explanation: There are three ways to climb to the top.

  1. 1 step + 1 step + 1 step

  2. 1 step + 2 steps

  3. 2 steps + 1 step

    *Notes

    • This is a Dynamic Programming problem that involves using the Fibonacci sequence. To find the # of ways to get to the nth step you need the n-1 and n-2 step
    • I tried using recursion for fibonacci sequence before but we ran out of time. This is because you are taking up extra time computing steps that you found previously. When things get big to something like n = 45, this becomes a problem. THIS MEANS USE MEMOIZATION
    • Instead I removed the recursion aspect all together and stored everything in an array of size n+1.
    • Set "base cases." arr[0] will be 1, arr[1] is 1, and arr[2] is 2. (arr[0] is 1 in order to make arr[2] = arr[1] + arr[0] true)
    • From i to n, if i is not 0, 1, 2, arr[n] equals arr[n-1] + arr[n-2]
    • Return n.
    • My first Dynamic Programming problem. Was getting out of bounds errors for a bit and had to manipulate size of the array to figure it out

EASY: [Problem 9]: Majority Element: Given an array nums of size n, return the majority element.

The majority element is the element that appears more than ⌊n / 2⌋ times. You may assume that the majority element always exists in the array.

Example 1:

Input: nums = [3,2,3] Output: 3 Example 2:

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

*Notes

  • I wanted to design this with a hashmap, so I decided we wanted the key to be the actual number in the array
  • And the value to be number of times the number was found
  • With the getOrDefault method, if the number was already present, we added 1 to the value of the key
  • If the key is greater than length of array/2, then we found our number. Return this number as the answer
  • If this was never accessed, return 0

EASY: [Problem 10]: Contains Duplicate: Given an integer array nums, return true if any value appears at least twice in the array, and return false if every element is distinct.

Example 1:

Input: nums = [1,2,3,1] Output: true Example 2:

Input: nums = [1,2,3,4] Output: false

*Notes

  • I immedietley related this to a Hashset problem due to us using similar logic in DFS/BFS
  • Use a HashSet of integers
  • From i to nums.length, if we've seen the number ( seen.contains(nums[i] ) return true
  • Otherwise, add the number to the hash set
  • Return false outside the loop

EASY: [Problem 11]: Merge Two Sorted Lists: You are given the heads of two sorted linked lists list1 and list2.

Merge the two lists into one sorted list. The list should be made by splicing together the nodes of the first two lists.

Return the head of the merged linked list. image *Notes

  • Return the head of the merged list, merged list in ascending order
  • So we need to make a dummy node that points to the first value
  • Create another ListNode 'current' that points to dummy
  • All we have to do is "rewire" the 'current' ListNode so it merges the lists in ascending order
  • While list1 is not null and list2 is not null
  • Check which list's node is greater
  • 'Current' listNode then equals the list with the smaller node
  • list1/list2 = list1/list2.next
    
  • Obviously have an else for list 1 and list 2 whichever one isn't greater
  • outside of if/else, current = current.next
  • outside of while loop:
  • if list1/list 2 is not null, current.next = list1/list2. This is because after we went through the lists, one of them was outstandingly greater/smaller. so we just append what's left onto current
  • return dummy.next, since dummy is just the first value of current

EASY: FIRST PYTHON SOLUTION: [Problem 11]: Valid Anagram: Given two strings s and t, return true if t is an anagram of s, and false otherwise.

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.

Example 1:

Input: s = "anagram", t = "nagaram" Output: true Example 2:

Input: s = "rat", t = "car" Output: false

*Notes

  • All we need to do is check if the FREQUENCY of characters in the first string matches that of the second string
  • We can do this by making 2 hashmaps (dictionaries in Python), one for string s and one for string t
  • Then go through all characters in s. The 'key' is the character and the value is Hashmap.get(char, 0) +1
  • Do the same for all characters in t.
  • In python when we do equals equals for the two hashmaps, it will essentially check if the frequencies are the same.
  • Function will return true if the frequencies in hashmaps are the same and false otherwise
  • Resubmitted using these notes and passed

EASY | FIRST DFS PROBLEM : [Problem 12]: Flood Fill: Given an m x n integer grid image where image[i][j] represents the pixel value of the image.

You are also given three integers sr, sc, and color. You should perform a flood fill on the image starting from the pixel image[sr][sc].

To perform a flood fill, consider the starting pixel, plus any pixels connected 4-directionally to the starting pixel of the same color as the starting pixel, plus any pixels connected 4-directionally to those pixels (also with the same color), and so on. Replace the color of all of the aforementioned pixels with color.

Return the modified image after performing the flood fill.

image

Input: image = [[1,1,1],[1,1,0],[1,0,1]], sr = 1, sc = 1, color = 2 Output: [[2,2,2],[2,2,0],[2,0,1]] Explanation: From the center of the image with position (sr, sc) = (1, 1) (i.e., the red pixel), all pixels connected by a path of the same color as the starting pixel (i.e., the blue pixels) are colored with the new color. Note the bottom corner is not colored 2, because it is not 4-directionally connected to the starting pixel.

Example 2:

Input: image = [[0,0,0],[0,0,0]], sr = 0, sc = 0, color = 0 Output: [[0,0,0],[0,0,0]] Explanation: The starting pixel is already colored 0, so no changes are made to the image.

/** *Notes

  • First DFS implementation, congrats!
  • A relativley easy problem once you understand what 4 directional means
  • This means anywhere up, down, left, right connected to our image[sr][sc]
  • We can only alter values that have the same value as the original image[sr][sc] and are related some way (either distantly or direct) 4 directionally
  • We need to create a DFS helper function as a result
  • First update the current color to the target. Then find if any left, right, up, down equal the oldTarget, and if so call dfs with these parameters
  • left means row-1, right means row+1, up means col-1, down means col+1. Make sure the bounds are valid as well
  • IMPORTANT: base case in main function is that if the image[sr][sc] equals target then we return image
  • My solution is DFS because once we find a valid 4 directional spot, we suspend all operations and explore this spot further and try to find its neighbors

*/

EASY: [Problem 13]: Search in a Binary Search Tree

You are given the root of a binary search tree (BST) and an integer val.

Find the node in the BST that the node's value equals val and return the subtree rooted with that node. If such a node does not exist, return null. image Input: root = [4,2,7,1,3], val = 2 Output: [2,1,3]

image Input: root = [4,2,7,1,3], val = 5 Output: []

*Notes

  • First Binary Search Tree Problem, congratulations
  • I felt like this was similar to a binary search.
  • If the root value is bigger than the target, call the function again with it's left child which would be smaller
  • Likewise if the root value is smaller than the target, call the function again with it's right child which would be bigger
  • Return root if none of these are true
  • This should all be inside a if(root!= null), after that, return Null which means the target isn't available.

FIRST MEDIUM: [Problem 14] : Lowest Common Ancestor of Binary Search Tree: Given a binary search tree (BST), find the lowest common ancestor (LCA) node of two given nodes in the BST.

According to the definition of LCA on Wikipedia: “The lowest common ancestor is defined between two nodes p and q as the lowest node in T that has both p and q as descendants (where we allow a node to be a descendant of itself).”

image Input: root = [6,2,8,0,4,7,9,null,null,3,5], p = 2, q = 8 Output: 6 Explanation: The LCA of nodes 2 and 8 is 6.

image Input: root = [6,2,8,0,4,7,9,null,null,3,5], p = 2, q = 4 Output: 2 Explanation: The LCA of nodes 2 and 4 is 2, since a node can be a descendant of itself according to the LCA definition.

  • This was very similar to the leetcode problem that searches for a target within a BST.
  • All we really had to change was to say if the root is greater than p and q, search in the left (where all values of BST less than root are on left)
  • if the root is less than p and q, search in the right(where all values of BST greater than root are on the right)
  • else, return the root.
  • This is a DFS implementation

EASY: [Problem 15]: Given an integer array nums, move all 0's to the end of it while maintaining the relative order of the non-zero elements.

Note that you must do this in-place without making a copy of the array.

Example 1:

Input: nums = [0,1,0,3,12] Output: [1,3,12,0,0] Example 2:

Input: nums = [0] Output: [0] *Notes

  • I saw the remove and append method looking through W3 schools
  • I saw that we can remove the current element if zero, and use append to send the zero to the end of the list
  • Maintains order of the other elements!

EASY: [Problem 16]: Given an integer array nums sorted in non-decreasing order, return an array of the squares of each number sorted in non-decreasing order.

Example 1:

Input: nums = [-4,-1,0,3,10] Output: [0,1,9,16,100] Explanation: After squaring, the array becomes [16,1,0,9,100]. After sorting, it becomes [0,1,9,16,100]. Example 2:

Input: nums = [-7,-3,2,3,11] Output: [4,9,9,49,121]

*Notes

  • The sort method in python makes it so much easier
  • Just square the elements (or multiply by itslef) for each i in nums
  • After you're done with the for loop, sort the array, and return the array

MEDIUM: [Problem 17]: Insert Interval: You are given an array of non-overlapping intervals intervals where intervals[i] = [starti, endi] represent the start and the end of the ith interval and intervals is sorted in ascending order by starti. You are also given an interval newInterval = [start, end] that represents the start and end of another interval.

Insert newInterval into intervals such that intervals is still sorted in ascending order by starti and intervals still does not have any overlapping intervals (merge overlapping intervals if necessary).

Return intervals after the insertion.

Note that you don't need to modify intervals in-place. You can make a new array and return it.

Example 1:

Input: intervals = [[1,3],[6,9]], newInterval = [2,5] Output: [[1,5],[6,9]] Example 2:

Input: intervals = [[1,2],[3,5],[6,7],[8,10],[12,16]], newInterval = [4,8] Output: [[1,2],[3,10],[12,16]] Explanation: Because the new interval [4,8] overlaps with [3,5],[6,7],[8,10]. *Notes

  • intervals is an array of arrays, representing the intervals in ascending order
  • obv we want to insert newInterval into these existing intervals, while maintining sorted ascending order
  • If newInterval ends before current interval starts, we can add newInterval to result
  • If newInterval starts before current interval ends, we can add current interval to new result array
  • else, we can merge overlaps by taking the minimum of 2 intervals and maximum of the overlapping intervals and make it the new interval
  • appending newInterval outside the loop is crucial
  • Problem is way less intimidating once looking at the test cases and examples honestly

EASY: [Problem 18]: Given two integer arrays nums1 and nums2, return an array of their intersection . Each element in the result must be unique and you may return the result in any order.

Example 1:

Input: nums1 = [1,2,2,1], nums2 = [2,2] Output: [2] Example 2:

Input: nums1 = [4,9,5], nums2 = [9,4,9,8,4] Output: [9,4] Explanation: [4,9] is also accepted.

*Notes

  • Could be pretty easy w/python intersect method but it feels like cheating
  • Create new list
  • loop through any of the lists
  • If the number is in the other list and not in newly created list
  • add number to newly created list
  • return list

EASY: [Problem 19]: Missing Number Given an array nums containing n distinct numbers in the range [0, n], return the only number in the range that is missing from the array.

Example 1:

Input: nums = [3,0,1] Output: 2 Explanation: n = 3 since there are 3 numbers, so all numbers are in the range [0,3]. 2 is the missing number in the range since it does not appear in nums. Example 2:

Input: nums = [0,1] Output: 2 Explanation: n = 2 since there are 2 numbers, so all numbers are in the range [0,2]. 2 is the missing number in the range since it does not appear in nums. Example 3:

Input: nums = [9,6,4,2,3,5,7,0,1] Output: 8 Explanation: n = 9 since there are 9 numbers, so all numbers are in the range [0,9]. 8 is the missing number in the range since it does not appear in nums.

*Notes

  • Had to add the +1 to len(nums) due to confusing test case
  • Python makes it easy by saying if i in range(len(nums)+1) not in nums, return i,
  • else we can return 0

MEDIUM: [Problem 20]: Three Sum 🤔: Given an integer array nums, return all the triplets [nums[i], nums[j], nums[k]] such that i != j, i != k, and j != k, and nums[i] + nums[j] + nums[k] == 0.

Notice that the solution set must not contain duplicate triplets.

Example 1:

Input: nums = [-1,0,1,2,-1,-4] Output: [[-1,-1,2],[-1,0,1]] Explanation: nums[0] + nums[1] + nums[2] = (-1) + 0 + 1 = 0. nums[1] + nums[2] + nums[4] = 0 + 1 + (-1) = 0. nums[0] + nums[3] + nums[4] = (-1) + 2 + (-1) = 0. The distinct triplets are [-1,0,1] and [-1,-1,2]. Notice that the order of the output and the order of the triplets does not matter.

*Notes

  • Super daunting problem at first and finally got around to getting it done, but needed help. So redo is needed
  • Recommended to do Two Sum II as well due to the sorting aspect in that problem

MEDIUM: [Problem 21]: Group Anagrams: Given an array of strings strs, group the anagrams together. You can return the answer in any order.

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.

Example 1:

Input: strs = ["eat","tea","tan","ate","nat","bat"] Output: [["bat"],["nat","tan"],["ate","eat","tea"]] Example 2:

Input: strs = [""] Output: [[""]] Example 3:

Input: strs = ["a"] Output: [["a"]]

*Notes

  • This took some time but I sat down with the problem and did it. Needed to surf the web to find some python dictionary methods as well
  • For each word, I sorted the word, and then made the array of char's a word again by using join
  • if the word wasnt in the map, we added it, with the key being the sorted word and value being the actual word
  • Else, we appended the value of the sorted word to add the current word
  • Then we went through the hashamp and appended to a resultant list for each value in the map
  • Returned the list of lists

MEDIUM: [Problem 22]: Top K Frequent Elements: Given an integer array nums and an integer k, return the k most frequent elements. You may return the answer in any order.

Example 1:

Input: nums = [1,1,1,2,2,3], k = 2 Output: [1,2] Example 2:

Input: nums = [1], k = 1 Output: [1]

*Notes

  • I made a hashamp, where the key is the actual number and value is how many times it appears So I used the setdefault method to do this as well
  • Made a new map that sorted the top frequent keys in ascending order using lambda x: newMap[x]
  • Then I went through the map until a kcount equaled k, and added the number to a list
  • returned the list
  • Due to sorting and the for loop I beleive this is O(n log n) time complexity

MEDIUM: [Problem 23]: Encode and Decode String: Design an algorithm to encode a list of strings to a single string. The encoded string is then decoded back to the original list of strings.

Please implement encode and decode

Example 1:

Input: ["neet","code","love","you"]

Output:["neet","code","love","you"] Example 2:

Input: ["we","say",":","yes"]

Output: ["we","say",":","yes"]

*Notes

  • The encode part wasn't bad in my eyes.
  • I used the neetcode hint that said to use a character like # to show we're about to start a new word, and the length of the word before the # so we can iterate properly
  • I then sent this combined string to the decode method
  • In decode, I made a variable i that tracked our spot in the combined string, and made a while loop while i was less than length of the string
  • I then made a pointer that said j = i
  • So then while s[j] != "#", I kept adding 1 to j, to represent the length of the string we needed
  • I then said the length was an integer from s[i:j]
  • I then appended from j +1 to length + j + 1
  • set i = j + 1 + length
  • return the list

MEDIUM: [Problem 24]: Longest Consecutive Sequence: 128. Longest Consecutive Sequence

Given an unsorted array of integers nums, return the length of the longest consecutive elements sequence.

You must write an algorithm that runs in O(n) time.

Example 1:

Input: nums = [100,4,200,1,3,2] Output: 4 Explanation: The longest consecutive elements sequence is [1, 2, 3, 4]. Therefore its length is 4. Example 2:

Input: nums = [0,3,7,2,5,8,4,6,0,1] Output: 9 *Notes

  • Did this problem alongside Eshaan. Got pretty close to figuring it out but needed some help
  • First set all the values in nums to a hashmap. This sorts the hashmap and gets rid of duplicates
  • For num in the hashmap, if the previous number exists, then we skip the iteration. Else, we found the beginning of the sequene.
  • Set j to 1 as we are at the beginning of the sequence
  • While num + j is in consecutive, add 1 to j.
  • This ensures that we only check for the next one. If it's not in consecutive, then j resets to 1 on the next iterations
  • Find the max between maximum and j, return maximum

MEDIUM: [Problem 25]: Product of Array Except Self:Given an integer array nums, return an array answer such that answer[i] is equal to the product of all the elements of nums except nums[i].

The product of any prefix or suffix of nums is guaranteed to fit in a 32-bit integer.

You must write an algorithm that runs in O(n) time and without using the division operation.

Example 1:

Input: nums = [1,2,3,4] Output: [24,12,8,6] Example 2:

Input: nums = [-1,1,0,-3,3] Output: [0,0,9,0,0]

*Horrible time and space complexity, even if it's O(n) 😭

  • Made a loop to get the prefixes and add it to prefix array (anything to the left of the element multiplied)
  • Did the same for the postfixes, but did for postfix array (anything to right side)
  • Also created a special case for if i =0, made sure I could multiply out everything except i
  • Separate for loop, if i was 0 I appended my special case variable
  • If i was the last element, I got the i-1th element from the prefix array
  • else, just append prefix i-1 and postifx i+1

MEDIUM: [Problem 26]: Two Sum II: Given a 1-indexed array of integers numbers that is already sorted in non-decreasing order, find two numbers such that they add up to a specific target number. Let these two numbers be numbers[index1] and numbers[index2] where 1 <= index1 < index2 <= numbers.length.

Return the indices of the two numbers, index1 and index2, added by one as an integer array [index1, index2] of length 2.

The tests are generated such that there is exactly one solution. You may not use the same element twice.

Your solution must use only constant extra space.

Example 1:

Input: numbers = [2,7,11,15], target = 9 Output: [1,2] Explanation: The sum of 2 and 7 is 9. Therefore, index1 = 1, index2 = 2. We return [1, 2]. Example 2:

Input: numbers = [2,3,4], target = 6 Output: [1,3] Explanation: The sum of 2 and 4 is 6. Therefore index1 = 1, index2 = 3. We return [1, 3]. Example 3:

Input: numbers = [-1,0], target = -1 Output: [1,2] Explanation: The sum of -1 and 0 is -1. Therefore index1 = 1, index2 = 2. We return [1, 2].

*Notes

  • Almost got there via binary search
  • Should make a bianry search but you don't need to find the middle.
  • Instead, add up left and right and see how it compares to target at the beginning of the while loop
  • Move left and right bounds by +- 1 based on relation to target
  • Pretty easy problem if you understand binary search

[MEDIUM]: Problem 27: Container With Most Water: You are given an integer array height of length n. There are n vertical lines drawn such that the two endpoints of the ith line are (i, 0) and (i, height[i]).

Find two lines that together with the x-axis form a container, such that the container contains the most water.

Return the maximum amount of water a container can store.

Notice that you may not slant the container.

Example 1:

Input: height = [1,8,6,2,5,4,8,3,7] Output: 49 Explanation: The above vertical lines are represented by array [1,8,6,2,5,4,8,3,7]. In this case, the max area of water (blue section) the container can contain is 49. Example 2:

Input: height = [1,1] Output: 1 *Notes

  • Honestly didnt seem like two pointers at first, but once you understand that we have to shift the left and right bounds depending on comparisons between left and right it makes sense
  • Is sort of like a 2d array, but not really.
  • Calculate the area using this: (right-left)* min(height[right], height[left])

[MEDIUM]: Problem 28: Longest Substring Without Repeating Characters: Given a string s, find the length of the longest substring without repeating characters.

Example 1:

Input: s = "abcabcbb" Output: 3 Explanation: The answer is "abc", with the length of 3. Example 2:

Input: s = "bbbbb" Output: 1 Explanation: The answer is "b", with the length of 1. Example 3:

Input: s = "pwwkew" Output: 3 Explanation: The answer is "wke", with the length of 3. Notice that the answer must be a substring, "pwke" is a subsequence and not a substring.

*Notes

  • I was able to do the pseudocode for this at the gym and honestly wasnt as bad as people made it out to be
  • When you understand that using a set is vital to this, as you can check whether or not an element is in the hashset or not, the problem becoems simpler
  • Calculate the length of the string using right-left+1
  • Remove the left most if you find the right most is already in the set. increment the left by 1 and move on
  • The right is added to the set if it's not in the set (the right is the current one, indicated by a for loop)

[MEDIUM]: Problem 29: Longest Repeating Character Replacement: You are given a string s and an integer k. You can choose any character of the string and change it to any other uppercase English character. You can perform this operation at most k times.

Return the length of the longest substring containing the same letter you can get after performing the above operations.

Example 1:

Input: s = "ABAB", k = 2 Output: 4 Explanation: Replace the two 'A's with two 'B's or vice versa. Example 2:

Input: s = "AABABBA", k = 1 Output: 4 Explanation: Replace the one 'A' in the middle with 'B' and form "AABBBBA". The substring "BBBB" has the longest repeating letters, which is 4. There may exists other ways to achieve this answer too.

*Notes

  • Pretty hard problem for me, interested to see the redo
  • right pointer starts at 0 and goes till end of range of string
  • left ptr also starts at 0, but only increments when the window size - the maximum of our values in a hashmap are greater than k
  • in this case, we also decrement one from the left side of the window, to remove that current occurence
  • at the end of each iteration, make a variable that tracks the window size (left-right+1), and that will be the return type
  • Also remember to add to the hashmap at the beginning of each iteration

[MEDIUM]: Problem 30: Permutation in String: Given two strings s1 and s2, return true if s2 contains a permutation of s1, or false otherwise.

In other words, return true if one of s1's permutations is the substring of s2.

Example 1:

Input: s1 = "ab", s2 = "eidbaooo" Output: true Explanation: s2 contains one permutation of s1 ("ba"). Example 2:

Input: s1 = "ab", s2 = "eidboaoo" Output: false

*Notes

  • My first time using Counter, which essentially maps a key to a value (AKA a letter to how many times it occurs)
  • I initialize the first variable as a counter for str1, but str2 is a counter from 0 to the size of str1
  • if the two 'str1' and 'str2' are equal, return true because it means they have the same combination of key value pairs, which represents a permutation
  • else, we need to adjust the window.
  • as a result, we reduce the frequency of s2[i] by 1
  • if there are no more occurences of s2[i] in the window (str2[s2[i]] == 0), we remove it from the dictionary
  • outside this if, we access the character in s2 that represents the next posiiton in the sliding window s2[len(s1)+i]
  • We slide the window over the entire string as a result

[MEDIUM]: Problem 31: Min Stack: Design a stack that supports push, pop, top, and retrieving the minimum element in constant time.

Implement the MinStack class:

MinStack() initializes the stack object. void push(int val) pushes the element val onto the stack. void pop() removes the element on the top of the stack. int top() gets the top element of the stack. int getMin() retrieves the minimum element in the stack. You must implement a solution with O(1) time complexity for each function.

Example 1:

Input ["MinStack","push","push","push","getMin","pop","top","getMin"] [[],[-2],[0],[-3],[],[],[],[]]

Output [null,null,null,null,-3,null,0,-2]

Explanation MinStack minStack = new MinStack(); minStack.push(-2); minStack.push(0); minStack.push(-3); minStack.getMin(); // return -3 minStack.pop(); minStack.top(); // return 0 minStack.getMin(); // return -2 *Notes

  • Not a terrible problem, just needed help getting started with def__init__ and intiialziing the two stacks as arrays
  • The caveat or hard part of the problem comes in retrieiving the minium in O(1) time. We can do this by having another stack that keeps track of the minimum
  • whenever you push, check to see if the val being pushed is less than the minimum (only if the length of the minstack exists)
  • if it is, make val equal to the end of the minStack
  • outside the if, append to the minStack the val
  • pop, top, annd getmin are one liners as a result of our minStack, Just remember to also pop from minstack when popping from normal stack

[MEDIUM] Problem 32: Evaluate Reverse Polish Notation:

[MEDIUM]: Problem 33: Daily Temperatures:

[MEDIUM] Problem 34: Car Fleet:

[MEDIUM] Problem 35: Search a 2D Matrix

[MEDIUM]: Problem 36: Koko Eating Bananas

[MEDIUM]: Problem 37: Find Minimum in Rotated Sorted Array

[EASY]: Problem 38: Invert Binary Tree

[EASY] Problem 39: Max Depth of Binary Tree

[EASY]: Porblem 40: Same Tree