1071. Greatest Common Divisor of Strings


Problem Description

The provided problem involves finding the largest string x that can divide two input strings str1 and str2. One string divides another if it can be repeated some number of times to obtain the other string. For example, if we have str1 = "abcabc" and str2 = "abc", then str2 divides str1 because we can concatenate str2 with itself twice to get str1. In other words, str1 is the result of appending str2 to itself multiple times. The challenge is to identify the largest common string that can divide both str1 and str2 in a similar fashion.

To illustrate further, if str1 is "ababab" and str2 is "abab", the string "ab" is the answer since it is the largest string that divides both str1 and str2.

Intuition

The underlying concept used in the provided solution is based on the mathematical idea of finding the greatest common divisor (GCD), but rather than numbers, we're dealing with strings. The key insight is that if there exists such a common string x that divides both str1 and str2, then the concatenation of str1 with str2 (str1 + str2) should be the same as the concatenation of str2 with str1 (str2 + str1). If this condition doesn't hold, it implies there is no such common string and the answer is an empty string.

Assuming the condition is true, we can find the length of the largest string x using the greatest common divisor of the lengths of str1 and str2. This is because the repeating pattern of string x must be a factor of both string lengths in order for it to divide both strings completely. Therefore, we find the GCD of the lengths of str1 and str2 and then return the substring of str1 up to that length. This substring is the largest common divider x.

Learn more about Math patterns.

Solution Approach

The solution provided takes advantage of Python's built-in gcd function from the [math](/problems/math-basics) library to calculate the greatest common divisor of the lengths of the two input strings str1 and str2. This is crucial for the solution, as the gcd represents the length of the largest string that can divide both strings, if such a string exists.

Here is how the solution unfolds:

  1. Check for Compatibility: The first step in the provided solution checks whether the strings are compatible to have a common divisor by concatenating str1 with str2 and comparing it to str2 concatenated with str1. This is facilitated by the operation if str1 + str2 != str2 + str1, which ensures that the pattern of characters in both strings is compatible for them to have a common divisor string. If the check fails, the two strings cannot have a common divisor and the function immediately returns an empty string ''.

  2. Find Length of the Common Divisor: If the strings pass the compatibility check, the next step is to determine the length of the largest common divisor string. This is where we use the gcd function, by calling n = gcd(len(str1), len(str2)). The gcd function takes the lengths of str1 and str2 and returns the greatest common divisor of these two numbers.

  3. Extract the Common Divisor String: With the length n obtained from the gcd function, the final step is to extract the common divisor string from str1. This is accomplished by the expression return str1[:n]. The [:n] slice operation on str1 returns the substring of str1 from the beginning up to the n-th character (exclusive), which is the required largest string that divides both str1 and str2.

To summarize, the algorithm consists of concatenation for compatibility checking and the greatest common divisor calculation, both of which are simple, efficient operations. The lack of any complex data structures and the use of a mathematical pattern of common division make this solution both elegant and effective. It leverages the fact that if a common divisor string exists, the repeating pattern must align with the gcd of the string lengths.

Ready to land your dream job?

Unlock your dream job with a 2-minute evaluator for a personalized learning plan!

Start Evaluator

Example Walkthrough

Let's use a small example to illustrate the solution approach. Suppose we have two strings str1 = "abab" and str2 = "ab". We need to find the largest string x that can divide both str1 and str2.

  1. Check for Compatibility: We first check if concatenating str1 with str2 is equal to str2 concatenated with str1.

    • str1 + str2 = "abab" + "ab" = "ababab"
    • str2 + str1 = "ab" + "abab" = "ababab"

    Since str1 + str2 is equal to str2 + str1, the two strings are compatible, and therefore, it's possible they have a common divisor string.

  2. Find Length of the Common Divisor: Next, we find the greatest common divisor of the lengths of str1 (4) and str2 (2).

    • gcd(4, 2) = 2

    The gcd tells us that the length of the largest string that possibly divides both str1 and str2 is 2.

  3. Extract the Common Divisor String: Since the gcd is 2, we take the first 2 characters from str1 to form our largest divisor string x.

    • str1[:2] = "ab"

    So, "ab" is the string we expect to divide both str1 and str2. To verify, we can see that:

    • str1 divided by "ab" is "abab" / "ab" = "ab", which is the repetition of "ab" twice.
    • str2 divided by "ab" is "ab" / "ab" = "", which is the repetition of "ab" once.

The largest string that can divide both str1 and str2 is "ab", which matches our result from the solution approach. The solution is elegant and efficient, leveraging simple concatenation and the mathematical concept of greatest common divisor to determine the existence and length of a common divisor string.

Solution Implementation

1from math import gcd
2
3class Solution:
4    def gcdOfStrings(self, str1: str, str2: str) -> str:
5        # Check if the concatenation of the two strings in different order results in the same string.
6        # This is required as the strings should be made of the same substrings for them to have a common divisor.
7        if str1 + str2 != str2 + str1:
8            # If they don't form the same string when concatenated in different orders, 
9            # there is no common divisor string and hence return an empty string.
10            return ''
11      
12        # Find the gcd of the lengths of the two strings.
13        # The gcd of the lengths will give us the maximum length the common divisor string can have.
14        length_gcd = gcd(len(str1), len(str2))
15      
16        # Return the substring from 0 to length_gcd from the first string, 
17        # which is the greatest common divisor string.
18        return str1[:length_gcd]
19
1class Solution {
2    // Function to find the greatest common divisor of lengths of two strings.
3    // This GCD can be used to find the longest substring that can construct
4    // the given strings by repeated concatenation.
5    public String gcdOfStrings(String str1, String str2) {
6        // Check if the two strings can be constructed from a common substring
7        // Only if str1+str2 equals str2+str1, they have a common divisor string
8        if (!(str1 + str2).equals(str2 + str1)) {
9            return ""; // If not, return an empty string as there is no common divisor
10        }
11        // Calculate the GCD of lengths of the two strings
12        int len = gcd(str1.length(), str2.length());
13        // The substring from the beginning of str1 with length 'len' is the gcd string
14        return str1.substring(0, len);
15    }
16
17    // Helper function to calculate the greatest common divisor (GCD) of two integers
18    // It uses the Euclidean algorithm to find the GCD
19    private int gcd(int a, int b) {
20        // Base case: if b is 0, then a is the GCD (as GCD(a, 0) = a)
21        // Recursive step: GCD(a, b) = GCD(b, a mod b)
22        return b == 0 ? a : gcd(b, a % b);
23    }
24}
25
1#include <algorithm> // include algorithm header for std::gcd
2
3class Solution {
4public:
5    // Function to find the greatest common divisor of strings str1 and str2
6    string gcdOfStrings(string str1, string str2) {
7        // check if concatenating the strings in both orders gives the same result
8        // this is required because two strings can only be multiples of each other
9        // if this condition is true
10        if (str1 + str2 != str2 + str1) {
11            return ""; // if they are not equivalent, return an empty string
12        }
13      
14        // calculate the greatest common divisor (GCD) of the sizes of the two strings
15        // std::gcd is available in C++17 and later. For C++14 and earlier, use a custom gcd function
16        int gcdValue = std::gcd(str1.size(), str2.size());
17      
18        // return the common divisor string which is the substring from start of str1 to its GCD length
19        return str1.substr(0, gcdValue);
20    }
21};
22
1// Function to calculate the greatest common divisor (GCD) of two numbers
2// Uses Euclidean algorithm
3function gcd(a: number, b: number): number {
4    while (b !== 0) {
5        let t = b;
6        b = a % b;
7        a = t;
8    }
9    return a;
10}
11
12// Function to find the greatest common divisor of strings str1 and str2
13function gcdOfStrings(str1: string, str2: string): string {
14    // Check if concatenating the strings in both orders gives the same result
15    // This is required because two strings can only be multiples of each other
16    // if this condition is true
17    if (str1 + str2 !== str2 + str1) {
18        return ""; // If they are not equivalent, return an empty string
19    }
20  
21    // Calculate the greatest common divisor (GCD) of the lengths of the two strings
22    const gcdValue = gcd(str1.length, str2.length);
23  
24    // Return the common divisor string which is the substring from start of str1 to its GCD length
25    return str1.substring(0, gcdValue);
26}
27

Time and Space Complexity

The code contains the gcdOfStrings method that finds the greatest common divisor (GCD) of lengths of two strings str1 and str2 to determine the largest string that can be repeatedly used to construct str1 and str2.

Time Complexity:

The time complexity of this function mainly depends on two operations: the string concatenation operation (str1 + str2 and str2 + str1) and the greatest common divisor computation (gcd(len(str1), len(str2))).

  1. String Concatenation: If str1 is of length n and str2 of length m, the concatenation will take O(n + m) time as each character from both strings need to be combined once.

  2. Checking String Equality: Comparing the concatenated strings takes O(n + m) time. If the strings differ, the method returns early.

  3. Greatest Common Divisor: The gcd function typically employs Euclid's algorithm, which has a time complexity of O(log(min(n, m))) where n and m are the lengths of the strings.

Therefore, the overall time complexity combines these operations resulting in O(n + m + log(min(n, m))). However, the dominating factor for large values of n and m will tend to be the concatenation and comparison, so we can approximate the time complexity as O(n + m).

Space Complexity:

The space complexity of the gcdOfStrings function can be analyzed as follows:

  1. Temporary Strings: The creation of concatenated strings str1 + str2 and str2 + str1 requires additional space of O(n + m).

  2. GCD Computation: The gcd function itself may use constant space, O(1), if the Euclidean algorithm is implemented in an iterative manner.

As such, no additional space that grows with the input size is required except for the string concatenation. Hence, the space complexity is O(n + m).

Learn more about how to find time and space complexity quickly using problem constraints.


Discover Your Strengths and Weaknesses: Take Our 2-Minute Quiz to Tailor Your Study Plan:
Question 1 out of 10

What does the following code do?

1def f(arr1, arr2):
2  i, j = 0, 0
3  new_arr = []
4  while i < len(arr1) and j < len(arr2):
5      if arr1[i] < arr2[j]:
6          new_arr.append(arr1[i])
7          i += 1
8      else:
9          new_arr.append(arr2[j])
10          j += 1
11  new_arr.extend(arr1[i:])
12  new_arr.extend(arr2[j:])
13  return new_arr
14
1public static List<Integer> f(int[] arr1, int[] arr2) {
2  int i = 0, j = 0;
3  List<Integer> newArr = new ArrayList<>();
4
5  while (i < arr1.length && j < arr2.length) {
6      if (arr1[i] < arr2[j]) {
7          newArr.add(arr1[i]);
8          i++;
9      } else {
10          newArr.add(arr2[j]);
11          j++;
12      }
13  }
14
15  while (i < arr1.length) {
16      newArr.add(arr1[i]);
17      i++;
18  }
19
20  while (j < arr2.length) {
21      newArr.add(arr2[j]);
22      j++;
23  }
24
25  return newArr;
26}
27
1function f(arr1, arr2) {
2  let i = 0, j = 0;
3  let newArr = [];
4  
5  while (i < arr1.length && j < arr2.length) {
6      if (arr1[i] < arr2[j]) {
7          newArr.push(arr1[i]);
8          i++;
9      } else {
10          newArr.push(arr2[j]);
11          j++;
12      }
13  }
14  
15  while (i < arr1.length) {
16      newArr.push(arr1[i]);
17      i++;
18  }
19  
20  while (j < arr2.length) {
21      newArr.push(arr2[j]);
22      j++;
23  }
24  
25  return newArr;
26}
27

Recommended Readings

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.

Coding Interview Strategies

Dive into our free, detailed pattern charts and company guides to understand what each company focuses on.

See Patterns
โ†
โ†‘๐Ÿช„