refactor: PermuteString (#5362)

This commit is contained in:
Alex Klymenko 2024-08-23 11:33:41 +02:00 committed by GitHub
parent 0301ecf1cb
commit d7b60be7d1
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
2 changed files with 102 additions and 35 deletions

View File

@ -1,51 +1,89 @@
package com.thealgorithms.strings; package com.thealgorithms.strings;
/* import java.util.HashSet;
Backtracking algorithm used in the program:- import java.util.Set;
>>Fix a character in the first position and swap the rest of the character with the first character. /**
Like in ABC, in the first iteration three strings are formed: ABC, BAC, and CBA by swapping A with * This class provides methods for generating all permutations of a given string using a backtracking algorithm.
A, B and C respectively. * <p>
>>Repeat step 1 for the rest of the characters like fixing second character B and so on. * The algorithm works as follows:
>>Now swap again to go back to the previous position. E.g., from ABC, we formed ABC by fixing B * <ol>
again, and we backtrack to the previous position and swap B with C. So, now we got ABC and ACB. * <li>Fix a character in the current position and swap it with each of the remaining characters.
>>Repeat these steps for BAC and CBA, to get all the permutations. * For example, for the string "ABC":
* <ul>
* <li>Fix 'A' at the first position: permutations are "ABC", "BAC", "CBA" (obtained by swapping 'A' with 'B' and 'C' respectively).</li>
* </ul>
* </li>
* <li>Repeat the process for the next character.
* For instance, after fixing 'B' in the second position:
* <ul>
* <li>For "BAC", the permutations include "BAC" and "BCA" (after swapping 'A' and 'C').</li>
* </ul>
* </li>
* <li>After generating permutations for the current position, backtrack by swapping the characters back to their original positions to restore the state.
* For example, after generating permutations for "ABC", swap back to restore "BAC" and continue with further permutations.</li>
* <li>Repeat the process for all characters to get all possible permutations.</li>
* </ol>
* </p>
*/ */
public final class PermuteString { public final class PermuteString {
private PermuteString() { private PermuteString() {
} }
// Function for swapping the characters at position I with character at position j /**
public static String swapString(String a, int i, int j) { * Generates all possible permutations of the given string.
char[] b = a.toCharArray(); *
char ch; * <p>This method returns a set containing all unique permutations of the input string. It leverages
ch = b[i]; * a recursive helper method to generate these permutations.
b[i] = b[j]; *
b[j] = ch; * @param str The input string for which permutations are to be generated.
return String.valueOf(b); * If the string is null or empty, the result will be an empty set.
* @return A {@link Set} of strings containing all unique permutations of the input string.
* If the input string has duplicate characters, the set will ensure that only unique permutations
* are returned.
*/
public static Set<String> getPermutations(String str) {
Set<String> permutations = new HashSet<>();
generatePermutations(str, 0, str.length(), permutations);
return permutations;
} }
public static void main(String[] args) { /**
String str = "ABC"; * Generates all permutations of the given string and collects them into a set.
int len = str.length(); *
System.out.println("All the permutations of the string are: "); * @param str the string to permute
generatePermutation(str, 0, len); * @param start the starting index for the current permutation
} * @param end the end index (length of the string)
* @param permutations the set to collect all unique permutations
// Function for generating different permutations of the string */
public static void generatePermutation(String str, int start, int end) { private static void generatePermutations(String str, int start, int end, Set<String> permutations) {
// Prints the permutations
if (start == end - 1) { if (start == end - 1) {
System.out.println(str); permutations.add(str);
} else { } else {
for (int i = start; i < end; i++) { for (int currentIndex = start; currentIndex < end; currentIndex++) {
// Swapping the string by fixing a character // Swap the current character with the character at the start index
str = swapString(str, start, i); str = swapCharacters(str, start, currentIndex);
// Recursively calling function generatePermutation() for rest of the characters // Recursively generate permutations for the remaining characters
generatePermutation(str, start + 1, end); generatePermutations(str, start + 1, end, permutations);
// Backtracking and swapping the characters again. // Backtrack: swap the characters back to their original positions
str = swapString(str, start, i); str = swapCharacters(str, start, currentIndex);
} }
} }
} }
/**
* Swaps the characters at the specified positions in the given string.
*
* @param str the string in which characters will be swapped
* @param i the position of the first character to swap
* @param j the position of the second character to swap
* @return a new string with the characters at positions i and j swapped
*/
private static String swapCharacters(String str, int i, int j) {
char[] chars = str.toCharArray();
char temp = chars[i];
chars[i] = chars[j];
chars[j] = temp;
return new String(chars);
}
} }

View File

@ -0,0 +1,29 @@
package com.thealgorithms.strings;
import static org.junit.jupiter.api.Assertions.assertEquals;
import java.util.Set;
import java.util.stream.Stream;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.MethodSource;
public class PermuteStringTest {
private static Stream<TestData> provideTestCases() {
return Stream.of(new TestData("ABC", Set.of("ABC", "ACB", "BAC", "BCA", "CAB", "CBA")), new TestData("AB", Set.of("AB", "BA")), new TestData("A", Set.of("A")), new TestData("AA", Set.of("AA")), new TestData("123", Set.of("123", "132", "213", "231", "312", "321")),
new TestData("aA", Set.of("aA", "Aa")), new TestData("AaB", Set.of("AaB", "ABa", "aAB", "aBA", "BAa", "BaA")), new TestData("!@", Set.of("!@", "@!")), new TestData("!a@", Set.of("!a@", "!@a", "a!@", "a@!", "@!a", "@a!")),
new TestData("ABCD", Set.of("ABCD", "ABDC", "ACBD", "ACDB", "ADBC", "ADCB", "BACD", "BADC", "BCAD", "BCDA", "BDAC", "BDCA", "CABD", "CADB", "CBAD", "CBDA", "CDAB", "CDBA", "DABC", "DACB", "DBAC", "DBCA", "DCAB", "DCBA")),
new TestData("A B", Set.of("A B", "AB ", " AB", " BA", "BA ", "B A")),
new TestData("abcd", Set.of("abcd", "abdc", "acbd", "acdb", "adbc", "adcb", "bacd", "badc", "bcad", "bcda", "bdac", "bdca", "cabd", "cadb", "cbad", "cbda", "cdab", "cdba", "dabc", "dacb", "dbac", "dbca", "dcab", "dcba")));
}
@ParameterizedTest
@MethodSource("provideTestCases")
void testPermutations(TestData testData) {
Set<String> actualPermutations = PermuteString.getPermutations(testData.input);
assertEquals(testData.expected, actualPermutations, "The permutations of '" + testData.input + "' are not correct.");
}
record TestData(String input, Set<String> expected) {
}
}