TheAlgorithms-Ruby/data_structures/arrays/remove_elements.rb

118 lines
2.4 KiB
Ruby
Raw Normal View History

# 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-16 05:41:43 +01:00
# Do not allocate extra space for another array,
# 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]
#
# Approach 1: Use `delete_if` Ruby method
#
# Time complexity: O(n)
#
def remove_elements(nums, val)
2021-09-03 22:24:58 +02:00
nums.delete_if { |num| num == val }
nums.length
end
2021-09-03 22:24:58 +02:00
puts remove_elements([3, 2, 2, 3], 3)
# => 2
2021-09-03 22:24:58 +02:00
puts remove_elements([0, 1, 2, 2, 3, 0, 4, 2], 2)
# => 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-09-03 22:24:58 +02:00
next unless num == val
nums.delete_at(i)
nums.unshift('removed')
result_length -= 1
shift_length += 1
end
nums.shift(shift_length)
result_length
end
2021-09-03 22:24:58 +02:00
puts remove_elements([3, 2, 2, 3], 3)
# => 2
2021-09-03 22:24:58 +02:00
puts remove_elements([0, 1, 2, 2, 3, 0, 4, 2], 2)
# => 5
2021-03-16 05:41:43 +01:00
#
2021-03-16 05:42:53 +01:00
# Approach 3: Two-pointers
2021-03-16 05:41:43 +01:00
#
2021-03-16 05:57:25 +01:00
# Complexity analysis
#
# Time complexity: O(n).
# Assume the array has a total of n elements,
2021-03-16 06:07:44 +01:00
# both pointer1 and pointer2 traverse at most 2n steps.
2021-03-16 05:57:25 +01: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-09-03 22:24:58 +02:00
puts remove_elements([3, 2, 2, 3], 3)
2021-03-16 06:00:08 +01:00
# => 2
2021-09-03 22:24:58 +02:00
puts remove_elements([0, 1, 2, 2, 3, 0, 4, 2], 2)
2021-03-16 06:00:08 +01:00
# => 5
2021-03-16 05:57:25 +01:00
#
# Approach 4: Two-pointers (Optimized)
#
2021-03-16 05:42:53 +01:00
# Complexity analysis
#
2021-03-16 06:07:44 +01:00
# Time complexity: O(n). Both pointer1 and pointer2 traverse at most n steps.
2021-03-16 05:42:53 +01:00
# In this approach, the number of assignment operations is equal to the
# number of elements to remove.
#
# Space complexity: O(1)
2021-03-16 05:41:43 +01:00
def remove_element(nums, val)
pointer1 = 0
pointer2 = nums.length
while pointer1 < pointer2
if nums[pointer1] == val
pointer2 -= 1
2021-03-16 05:50:51 +01:00
nums[pointer1] = nums[pointer2]
2021-03-16 05:41:43 +01:00
else
pointer1 += 1
end
end
pointer1
end
2021-09-03 22:24:58 +02:00
puts remove_elements([3, 2, 2, 3], 3)
2021-03-16 05:41:43 +01:00
# => 2
2021-09-03 22:24:58 +02:00
puts remove_elements([0, 1, 2, 2, 3, 0, 4, 2], 2)
2021-03-16 05:41:43 +01:00
# => 5