Given a string containing alphanumeric and special characters, determine if it's a palindrome, ignoring special characters and case. Can you do this in-place (without using extra space)? For example, "12@#.Race@#car" is a palindrome. How would you approach this?
Given a string containing alphanumeric and special characters, determine if it's a palindrome, ignoring special characters and case. The goal is to achieve this in-place, meaning without using extra space.
The approach involves using two pointers, one starting from the beginning of the string and the other from the end. We move these pointers towards each other, skipping non-alphanumeric characters and ignoring case, until they meet in the middle. If at any point the characters at the two pointers don't match, the string is not a palindrome.
def is_palindrome_in_place(s):
left = 0
right = len(s) - 1
while left < right:
# Move left pointer to the next alphanumeric character
while left < right and not s[left].isalnum():
left += 1
# Move right pointer to the previous alphanumeric character
while left < right and not s[right].isalnum():
right -= 1
# Compare characters (case-insensitive)
if s[left].lower() != s[right].lower():
return False
left += 1
right -= 1
return True
# Example usage
string1 = "12@#.Race@#car"
string2 = "A man, a plan, a canal: Panama"
string3 = "hello"
print(f"{string1}: {is_palindrome_in_place(string1)}")
print(f"{string2}: {is_palindrome_in_place(string2)}")
print(f"{string3}: {is_palindrome_in_place(string3)}")
left
and right
, are initialized to the start and end of the string, respectively.while
loop continues as long as left
is less than right
.while
loops increment left
and decrement right
until they point to alphanumeric characters.s[left]
and s[right]
are converted to lowercase and compared. If they are not equal, the function returns False
.left
is incremented and right
is decremented to check the next pair of characters.True
.The algorithm iterates through the string once, with the left
and right
pointers moving towards the middle. In the worst case, each pointer might have to traverse the entire string. Even though there are nested while loops, each character is visited a maximum of 4 times, so the time complexity is O(n), where n is the length of the string.
The algorithm uses a constant amount of extra space for the left
and right
pointers. Therefore, the space complexity is O(1), indicating that it operates in-place.
None
or undefined to prevent errors.def is_palindrome_in_place_safe(s):
if s is None:
return False # Or raise an exception, depending on requirements
left = 0
right = len(s) - 1
while left < right:
# Move left pointer to the next alphanumeric character
while left < right and not s[left].isalnum():
left += 1
# Move right pointer to the previous alphanumeric character
while left < right and not s[right].isalnum():
right -= 1
# Compare characters (case-insensitive)
if s[left].lower() != s[right].lower():
return False
left += 1
right -= 1
return True
# Example with None
string4 = None
print(f"{string4}: {is_palindrome_in_place_safe(string4)}")