2021-03-12 18:51:41 -08:00
|
|
|
# Arrays - Remove Elements
|
|
|
|
#
|
|
|
|
# Given an array nums and a value val, remove all instances of that value in-place and return the new length.
|
2021-03-15 21:41:43 -07:00
|
|
|
# Do not allocate extra space for another array,
|
2021-03-12 18:51:41 -08:00
|
|
|
# you must do this by modifying the input array in-place with O(1) extra memory.
|
|
|
|
# The order of elements can be changed. It doesn't matter what you leave beyond the new length.
|
|
|
|
#
|
|
|
|
# Example
|
|
|
|
#
|
|
|
|
# Input: nums = [3,2,2,3], val = 3
|
|
|
|
# Output: 2, nums = [2,2]
|
|
|
|
#
|
|
|
|
# Input: nums = [0,1,2,2,3,0,4,2], val = 2
|
|
|
|
# Output: 5, nums = [0,1,4,0,3]
|
|
|
|
|
|
|
|
#
|
2021-03-15 09:52:22 -07:00
|
|
|
# Approach 1: Use `delete_if` Ruby method
|
|
|
|
#
|
|
|
|
# Time complexity: O(n)
|
|
|
|
#
|
|
|
|
def remove_elements(nums, val)
|
|
|
|
nums.delete_if{ |num| num == val }
|
|
|
|
nums.length
|
|
|
|
end
|
|
|
|
|
|
|
|
puts remove_elements([3,2,2,3], 3)
|
|
|
|
# => 2
|
|
|
|
puts remove_elements([0,1,2,2,3,0,4,2], 2)
|
2021-03-15 10:08:31 -07:00
|
|
|
# => 5
|
|
|
|
|
|
|
|
#
|
|
|
|
# Approach 2: Use `delete_at`, `unshift`, and `shift` Ruby method
|
|
|
|
#
|
|
|
|
# Time complexity: O(n)
|
|
|
|
#
|
|
|
|
def remove_elements(nums, val)
|
|
|
|
result_length = nums.length
|
|
|
|
shift_length = 0
|
|
|
|
nums.each_with_index do |num, i|
|
2021-03-15 21:41:43 -07:00
|
|
|
if num == val
|
2021-03-15 10:08:31 -07:00
|
|
|
nums.delete_at(i)
|
|
|
|
nums.unshift('removed')
|
|
|
|
result_length -=1
|
|
|
|
shift_length += 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
nums.shift(shift_length)
|
|
|
|
result_length
|
|
|
|
end
|
|
|
|
|
|
|
|
puts remove_elements([3,2,2,3], 3)
|
|
|
|
# => 2
|
|
|
|
puts remove_elements([0,1,2,2,3,0,4,2], 2)
|
|
|
|
# => 5
|
2021-03-15 21:41:43 -07:00
|
|
|
|
|
|
|
#
|
2021-03-15 21:42:53 -07:00
|
|
|
# Approach 3: Two-pointers
|
2021-03-15 21:41:43 -07:00
|
|
|
#
|
|
|
|
|
2021-03-15 21:57:25 -07:00
|
|
|
# Complexity analysis
|
|
|
|
#
|
|
|
|
# Time complexity: O(n).
|
|
|
|
# Assume the array has a total of n elements,
|
2021-03-15 22:07:44 -07:00
|
|
|
# both pointer1 and pointer2 traverse at most 2n steps.
|
2021-03-15 21:57:25 -07:00
|
|
|
#
|
|
|
|
# Space complexity: O(1).
|
|
|
|
|
|
|
|
def remove_element(nums, val)
|
|
|
|
pointer1 = 0
|
|
|
|
|
|
|
|
nums.each_with_index do |num, pointer2|
|
|
|
|
if val != num
|
|
|
|
nums[pointer1] = nums[pointer2]
|
|
|
|
pointer1 += 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
pointer1
|
|
|
|
end
|
|
|
|
|
2021-03-15 22:00:08 -07:00
|
|
|
puts remove_elements([3,2,2,3], 3)
|
|
|
|
# => 2
|
|
|
|
puts remove_elements([0,1,2,2,3,0,4,2], 2)
|
|
|
|
# => 5
|
|
|
|
|
2021-03-15 21:57:25 -07:00
|
|
|
#
|
|
|
|
# Approach 4: Two-pointers (Optimized)
|
|
|
|
#
|
|
|
|
|
2021-03-15 21:42:53 -07:00
|
|
|
# Complexity analysis
|
|
|
|
#
|
2021-03-15 22:07:44 -07:00
|
|
|
# Time complexity: O(n). Both pointer1 and pointer2 traverse at most n steps.
|
2021-03-15 21:42:53 -07:00
|
|
|
# In this approach, the number of assignment operations is equal to the
|
|
|
|
# number of elements to remove.
|
|
|
|
#
|
|
|
|
# Space complexity: O(1)
|
|
|
|
|
2021-03-15 21:41:43 -07:00
|
|
|
def remove_element(nums, val)
|
|
|
|
pointer1 = 0
|
|
|
|
pointer2 = nums.length
|
|
|
|
|
|
|
|
while pointer1 < pointer2
|
|
|
|
if nums[pointer1] == val
|
|
|
|
pointer2 -= 1
|
2021-03-15 21:50:51 -07:00
|
|
|
nums[pointer1] = nums[pointer2]
|
2021-03-15 21:41:43 -07:00
|
|
|
else
|
|
|
|
pointer1 += 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
pointer1
|
|
|
|
end
|
|
|
|
|
|
|
|
puts remove_elements([3,2,2,3], 3)
|
|
|
|
# => 2
|
|
|
|
puts remove_elements([0,1,2,2,3,0,4,2], 2)
|
|
|
|
# => 5
|