2217. Find Palindrome With Fixed Length
Problem Description
The task here is to find certain special numbers called "positive palindromes." A positive palindrome is a number that reads the same both forward and backward, and it does not have any leading zeros. Given two inputs: an array of integers named queries
and a positive integer intLength
, our goal is to determine the intLength
-digit palindrome corresponding to each query.
For each integer in queries
, we interpret it as the queries[i]-th
smallest positive palindrome of length intLength
. If this palindrome exists, we add it to our answer array; otherwise, we append -1
to indicate there's no such palindrome for that query.
If we think about the nature of palindromes, we can realize that for a palindrome of a given length, the first half of the number dictates the second half due to the mirror-like property of palindromes. Therefore, finding a palindrome can be done by constructing its first half and then mirroring it to create the second half.
The problem requires us to handle the fact that palindromes of even and odd lengths behave slightly differently: an odd-length palindrome will have a single digit in the middle that isn't mirrored.
Intuition
The solution utilizes the pattern that palindromes of a certain length can be constructed by taking a base number and mirroring its digits. The base number is essentially the first half of the palindrome.
- For
intLength
that is odd, the middle digit is part of the base. - For
intLength
that is even, the base directly mirrors itself to form the whole palindrome.
To construct the smallest intLength
-digit palindrome, we need to start with the smallest base possible that, once mirrored, forms a palindrome of that length. This smallest base number is 10^(l-1)
where l
is the length of the base. The base itself is half of intLength
: l = (intLength + 1) // 2
.
The scope of possible bases ranges from this smallest base to 10^l - 1
. This is because 10^l
would result in a palindrome exceeding intLength
digits, violating the problem constraints.
With this understanding, the solution consists of the following steps:
-
Calculate the length
l
of the base number for the palindrome. IfintLength
is even,l
is half ofintLength
. IfintLength
is odd,l
is half ofintLength
, rounded up. -
Determine the starting point (
start
) and the endpoint (end
) for possible bases—the starting point being10^(l-1)
and the endpoint being10^l - 1
. -
Iterate over each query in
queries
:- Calculate a tentative palindrome base
v
by adding the 0-based index of the query (q - 1
) to the starting point. - If this base is greater than the range's endpoint (
end
), append-1
to the answers array (ans
), signifying no such palindrome exists. - Otherwise, create a string (
s
) for the first half of the palindrome (which isv
), mirror it, and append it to itself. For oddintLength
, ignore the last digit of the mirrored portion to prevent duplication of the middle digit. - Convert the resulting string back to an integer and append it to
ans
.
- Calculate a tentative palindrome base
-
Return the completed
ans
list, which now contains thequeries[i]
-th smallest palindrome or-1
for each query tested.
Learn more about Math patterns.
Solution Approach
The solution approach leverages simple integer and string operations to generate palindromes of a specific length. Here's a more detailed breakdown of the implementation:
-
Determine Base Length (
l
):- The first step in the algorithm involves calculating the length of the base number
l
, from which the entire palindrome can be constructed. This length is found by dividingintLength
by 2 and rounding up if necessary. It uses the right shift operator>> 1
which is equivalent to dividing by 2.
1l = (intLength + 1) >> 1
- The first step in the algorithm involves calculating the length of the base number
-
Define Start and End Range for Bases:
- To form palindromes of
intLength
, the starting point is the smallest possible positive integer of half that length, which is10**(l - 1)
. The endpoint is the largest integer of that half-length,10**l - 1
. These define the range of numbers that can be the first half of a palindrome.
1start, end = 10 ** (l - 1), 10**l - 1
- To form palindromes of
-
Iterate Over Queries and Construct Palindromes:
- The main logic happens in a loop iterating over each query. For each query, a base value
v
for the palindrome is created by adding the query's 0-based index to the start of the range. Then the solution checks whetherv
exceeds theend
, in which case it adds-1
to the answer array, indicating that no such palindrome exists within the given length constraint.
1v = start + q - 1 2if v > end: 3 ans.append(-1)
- The main logic happens in a loop iterating over each query. For each query, a base value
-
Constructing the Full Palindrome String:
- When a valid base
v
is found, it is converted to a strings
. To construct the whole palindrome,s
is concatenated with its reverse (s[::-1]
). For palindromes of an odd length, the middle character should not be duplicated, so the slice[intLength % 2:]
trims the first character from the reversed string before concatenation ifintLength
is odd.
1s += s[::-1][intLength % 2 :]
- When a valid base
-
Finalize and Return the Answer:
- After constructing the full palindrome string for each query, it is converted back to an integer and appended to the answer array
ans
. Once all queries are processed,ans
is returned as the result containing the requested palindromes or-1
when no such palindrome exists.
1ans.append(int(s))
- After constructing the full palindrome string for each query, it is converted back to an integer and appended to the answer array
The algorithm effectively uses string manipulation to leverage the inherent symmetry of palindromes, constructing them in an optimal manner by only dealing with half the number and mirroring it to get the full length. This means the time complexity is primarily determined by the number of queries and the computational complexity of string manipulation, both of which are managed efficiently within the provided problem constraints.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Let's walk through a small example to illustrate the solution approach with queries = [1, 2, 4]
and intLength = 3
.
-
Determine the Base Length (
l
):- Since
intLength
is 3, which is odd, we calculatel
as(3 + 1) >> 1
, yieldingl = 2
.
- Since
-
Define Start and End Range for Bases:
- The starting point
start
is10**(2 - 1)
, which equals10
. - The endpoint
end
is10**2 - 1
, which equals99
.
- The starting point
-
Iterate Over Queries and Construct Palindromes:
- The algorithm will loop over the queries
[1, 2, 4]
to find the corresponding palindromes.
- The algorithm will loop over the queries
-
Constructing Palindromes:
-
For the first query (1):
- We calculate
v
asstart + 1 - 1
, which is10
. v
is within the range, so we proceed to construct the palindrome.- The string representation of
v
is'10'
, and becauseintLength
is 3 (odd), we don't repeat the middle digit when reversing. - The palindrome becomes
'101'
, and we append the integer101
to our answerans
.
- We calculate
-
For the second query (2):
- We calculate
v
asstart + 2 - 1
, which is11
. - This is also within range, so the palindrome would be
'111'
, and we add111
toans
.
- We calculate
-
For the fourth query (4):
- We calculate
v
asstart + 4 - 1
, which is13
. - This base is not greater than
end
, so we construct the palindrome'131'
and add131
toans
.
- We calculate
-
-
Finalize and Return the Answer:
- Since there is no third query, we do not perform any action for it as it was not given in
queries
. - We also note that no queries in this example exceed our range, so there is no need to add
-1
at any point. - Our final answer
ans
is[101, 111, 131]
.
- Since there is no third query, we do not perform any action for it as it was not given in
The algorithm has successfully determined the queries[i]
-th smallest positive palindrome of length intLength
for each provided query. The palindromes are [101, 111, 131]
for the 1st
, 2nd
, and 4th
smallest palindromes of length 3, respectively.
Solution Implementation
1from typing import List
2
3class Solution:
4 def kth_palindrome(self, queries: List[int], int_length: int) -> List[int]:
5 # Calculate the number of digits in the first half of the palindrome
6 half_length = (int_length + 1) >> 1
7
8 # Define the start and end of the range for the first half of the palindrome
9 start = 10 ** (half_length - 1)
10 end = (10 ** half_length) - 1
11
12 # Initialize an empty list to store the answers
13 answers = []
14
15 # Iterate over each query to find the k-th palindrome
16 for query in queries:
17 # Calculate the value in the first half by offsetting the start with the query index
18 value = start + query - 1
19
20 # If the value exceeds the end boundary, the palindrome doesn't exist
21 if value > end:
22 answers.append(-1)
23 continue
24
25 # Convert the first half to a string
26 half_str = str(value)
27
28 # Construct the full palindrome by concatenating the first half and its reverse
29 # If the integer length is odd, skip the last digit of the reversed half
30 palindrome = half_str + half_str[::-1][int_length % 2:]
31
32 # Append the palindrome to the answers list, converting it back to an integer
33 answers.append(int(palindrome))
34
35 # Return the list of answers
36 return answers
37
1class Solution {
2
3 // Function to find kth palindromic number with specified length
4 public long[] kthPalindrome(int[] queries, int intLength) {
5 // Initialize an array for the answers with the same length as the queries array
6 long[] palindromes = new long[queries.length];
7 // Calculate the length of the first half of the palindromes
8 int halfLength = (intLength + 1) >> 1;
9 // Determine the start position of palindromes
10 long startNum = (long) Math.pow(10, halfLength - 1);
11 // Determine the end position of palindromes
12 long endNum = (long) Math.pow(10, halfLength) - 1;
13
14 // Iterate through all the queries
15 for (int i = 0; i < queries.length; ++i) {
16 // Calculate the value for current palindrome
17 long value = startNum + queries[i] - 1;
18
19 // If the value exceeds the upper bound of halfLength palindrome, set the result as -1
20 if (value > endNum) {
21 palindromes[i] = -1;
22 continue;
23 }
24
25 // Convert the number to a string
26 String halfPalindrome = Long.toString(value);
27 // Generate the full palindrome string
28 String fullPalindrome = halfPalindrome +
29 new StringBuilder(halfPalindrome).reverse().substring(intLength % 2);
30
31 // Add to the results after parsing the string back to a long
32 palindromes[i] = Long.parseLong(fullPalindrome);
33 }
34 // Return the array containing all the palindrome numbers
35 return palindromes;
36 }
37}
38
1#include <vector>
2#include <cmath>
3#include <algorithm>
4#include <string>
5
6class Solution {
7public:
8 // This function finds the kth smallest palindrome of a given length.
9 std::vector<long long> kthPalindrome(std::vector<int>& queries, int intLength) {
10 // Calculate the length of half of the palindrome
11 // because a palindrome reads the same backward as forward.
12 int halfLength = (intLength + 1) >> 1; // equivalent to (intLength + 1) / 2 but faster
13
14 // Starting value for half of the palindrome (e.g., 100...0 with halfLength number of digits)
15 long long start = std::pow(10, halfLength - 1);
16
17 // Ending value for half of the palindrome (e.g., 999...9 with halfLength number of digits)
18 long long end = std::pow(10, halfLength) - 1;
19
20 // Vector to store the resulting palindromes.
21 std::vector<long long> palindromes;
22
23 // Loop through each query to find the respective palindrome.
24 for (int q : queries) {
25 // Calculate the value for this query by offsetting from the starting value.
26 long long value = start + q - 1;
27
28 // If the calculated number exceeds the largest number with halfLength digits, it is not possible
29 // to generate the palindrome, so we add -1.
30 if (value > end) {
31 palindromes.push_back(-1);
32 continue;
33 }
34
35 // Convert the number to the first half of the palindrome.
36 std::string firstHalf = std::to_string(value);
37
38 // Prepare the second half by reversing the first half.
39 std::string secondHalf = firstHalf;
40 std::reverse(secondHalf.begin(), secondHalf.end());
41
42 // If the palindrome is of odd length, we omit the last digit in the second half.
43 if (intLength % 2 == 1) {
44 secondHalf = secondHalf.substr(1);
45 }
46
47 // Combine both halves to form the full palindrome.
48 std::string fullPalindromeStr = firstHalf + secondHalf;
49
50 // Convert the string to a long long and add to the results.
51 palindromes.push_back(std::stoll(fullPalindromeStr));
52 }
53
54 // Return the vector containing all the found palindromes.
55 return palindromes;
56 }
57};
58
1function kthPalindrome(queries: number[], intLength: number): number[] {
2 // Determine if the integer's length is odd.
3 const isOdd = intLength % 2 === 1;
4
5 // Calculate the base number, which is the smallest number of the given integer length.
6 // It's used to generate palindromes by prefixing it to its reverse (or almost reverse if the length is odd).
7 const baseNumber = 10 ** (Math.floor(intLength / 2) + (isOdd ? 1 : 0) - 1);
8
9 // Calculate the maximum valid query based on the integer length, used for bounds checking.
10 const maxQueryValue = baseNumber * 9;
11
12 // Map each query to its corresponding palindrome or -1 if the query is out of bounds.
13 return queries.map(query => {
14 if (query > maxQueryValue) {
15 return -1; // Query value exceeds the upper limit of possible palindromes.
16 }
17
18 // Calculate the palindrome's non-reversed part.
19 const nonReversedPartOfPalindrome = baseNumber + query - 1;
20
21 // Convert the number to a string, reverse it, and slice off the first digit if the length is odd.
22 const reversedPartOfPalindrome = nonReversedPartOfPalindrome
23 .toString()
24 .split('')
25 .reverse()
26 .slice(isOdd ? 1 : 0)
27 .join('');
28
29 // Construct the full palindrome and return it as a number.
30 const palindrome = Number(nonReversedPartOfPalindrome.toString() + reversedPartOfPalindrome);
31 return palindrome;
32 });
33}
34
Time and Space Complexity
Time Complexity
The time complexity of the given code can be broken down into a couple of key operations that occur within the loop running once for each element in queries
.
-
Calculation of the half-length: This is done only once before the loop, taking a constant time, so it does not affect the total time complexity.
-
The loop: The main loop runs once for each query in
queries
, hence, if there aren
queries, the loop runsn
times. -
String operations within the loop:
- Creating the
s
string: This involves creating a string representation of an integer which isO(L)
whereL
is the number of digits in the integer. - String slicing and concatenation: The slicing
s[::-1]
takesO(L)
time and the concatenations[::-1][intLength % 2 :]
also takesO(L)
time, as the length of the slice is proportional toL
. - Creating the
ans
listappend
operation: Appending to the list isO(1)
.
- Creating the
Since L
(length of the palindrome) is at most half of intLength
and intLength
itself is a constant with respect to the length of queries
, the operations inside the loop that depend on intLength
can be considered to occur in constant time as well. Thus, the total time complexity for the loop can be approximated as O(n)
where n
is the number of queries.
Space Complexity
For space complexity, we are mostly concerned with additional space that the program uses aside from the input and output.
-
The
ans
list: This list increases proportionally with the number of queriesn
, so the space complexity contribution here isO(n)
. -
Temporary variables
v
ands
: These are used for each query and do not depend on the number of queries. They do not increase the space complexity beyondO(n)
.
Therefore, the overall space complexity is O(n)
where n
is the number of queries.
Learn more about how to find time and space complexity quickly using problem constraints.
What's the output of running the following function using input [30, 20, 10, 100, 33, 12]
?
1def fun(arr: List[int]) -> List[int]:
2 import heapq
3 heapq.heapify(arr)
4 res = []
5 for i in range(3):
6 res.append(heapq.heappop(arr))
7 return res
8
1public static int[] fun(int[] arr) {
2 int[] res = new int[3];
3 PriorityQueue<Integer> heap = new PriorityQueue<>();
4 for (int i = 0; i < arr.length; i++) {
5 heap.add(arr[i]);
6 }
7 for (int i = 0; i < 3; i++) {
8 res[i] = heap.poll();
9 }
10 return res;
11}
12
1class HeapItem {
2 constructor(item, priority = item) {
3 this.item = item;
4 this.priority = priority;
5 }
6}
7
8class MinHeap {
9 constructor() {
10 this.heap = [];
11 }
12
13 push(node) {
14 // insert the new node at the end of the heap array
15 this.heap.push(node);
16 // find the correct position for the new node
17 this.bubble_up();
18 }
19
20 bubble_up() {
21 let index = this.heap.length - 1;
22
23 while (index > 0) {
24 const element = this.heap[index];
25 const parentIndex = Math.floor((index - 1) / 2);
26 const parent = this.heap[parentIndex];
27
28 if (parent.priority <= element.priority) break;
29 // if the parent is bigger than the child then swap the parent and child
30 this.heap[index] = parent;
31 this.heap[parentIndex] = element;
32 index = parentIndex;
33 }
34 }
35
36 pop() {
37 const min = this.heap[0];
38 this.heap[0] = this.heap[this.size() - 1];
39 this.heap.pop();
40 this.bubble_down();
41 return min;
42 }
43
44 bubble_down() {
45 let index = 0;
46 let min = index;
47 const n = this.heap.length;
48
49 while (index < n) {
50 const left = 2 * index + 1;
51 const right = left + 1;
52
53 if (left < n && this.heap[left].priority < this.heap[min].priority) {
54 min = left;
55 }
56 if (right < n && this.heap[right].priority < this.heap[min].priority) {
57 min = right;
58 }
59 if (min === index) break;
60 [this.heap[min], this.heap[index]] = [this.heap[index], this.heap[min]];
61 index = min;
62 }
63 }
64
65 peek() {
66 return this.heap[0];
67 }
68
69 size() {
70 return this.heap.length;
71 }
72}
73
74function fun(arr) {
75 const heap = new MinHeap();
76 for (const x of arr) {
77 heap.push(new HeapItem(x));
78 }
79 const res = [];
80 for (let i = 0; i < 3; i++) {
81 res.push(heap.pop().item);
82 }
83 return res;
84}
85
Recommended Readings
Math for Technical Interviews How much math do I need to know for technical interviews The short answer is about high school level math Computer science is often associated with math and some universities even place their computer science department under the math faculty However the reality is that you
Patterns The Shortest Path Algorithm for Coding Interviews The goal of AlgoMonster is to help you get a job in the shortest amount of time possible in a data driven way We compiled datasets of tech interview problems and broke them down by patterns This way we can determine the
Recursion Recursion is one of the most important concepts in computer science Simply speaking recursion is the process of a function calling itself Using a real life analogy imagine a scenario where you invite your friends to lunch https algomonster s3 us east 2 amazonaws com recursion jpg You first
Got a question? Ask the Monster Assistant anything you don't understand.
Still not clear? Submit the part you don't understand to our editors. Or join our Discord and ask the community.