Processing math: 100%

Monday, October 1, 2018

Permutations of a String - Unique Characters and Duplicates

REFERENCE @ Geek For Geeks

Permutation

Permutation of a set is simply a different ordering of the elements. For the given string, we can order the characters into different locations producing different string. All such strings constitute the permutation of that string.
Example: Permutation of a string XYZ are:

XYZ XZY YXZ YZX ZXY ZYX
For given string of length n there are total of n! permutations.

Algorithm - For String with Unique Characters

As explained HERE, following image explains how we can compute the permutations of given string.
In this algorithm,
  • For each character in the input String:
    1. Select the character and keep it fixed.
    2. Compute permutations of remaining n1 characters.
    3. Finally append all those permutations of n1 characters to the character selected in the Step 1.

Algorithm - For String with Duplicate Characters

Algorithms described above will also work with strings with duplicate characters. However, the algorithm will also produce duplicate results. So a simple solution will be to check for unique solutions, probably using Hashing, in the above solution.

Problem

The problem with this solution is, for a string with two many duplicate characters, it will end up computing all n! solutions while there are very few number of solutions.
Example: String AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA has only one permutation which the input string itself.

But the above algorithm will be doing a lot of unnecessary computation.

Solution

So for string with duplicate entries, we can write efficient solution. For string with all unique characters, its performance will still become n!.

The solution is very much similar to the Coin Exchange Problem.
In this algorithm,
  1. We can create Frequency table which keeps count of all unique characters in the string.
  2. For each unique character if it count is Greater Than 0:
    1. Append the character to Current Permutation. When length of the Current Permutation is equal to the length of input string, we have one solution.
    2. Decrease count by 1.
    3. Recurse through the same updated Map.
    4. Reset or Increase the count by 1.

Implementation (Both Method)

Following is the implementation for both the method.
/* File: Permutations.java
* Created: 10/1/2018
* Author: Sabbir Manandhar
*
* Copyright (c) 2018 Hogwart Inc.
*/
import java.util.*;
//=======================================================================================
/**
* @author Sabbir Manandhar manandharsabbir@gmail.com
* @version 1.0
*/
public class Permutations {
public static void main(String[] args) {
List<String> permutations = permutationWithoutDuplicates("ABCD");
for (String permutation : permutations)
System.out.println(permutation);
System.out.println();
permutations = permuteWithDuplicates("AAABB");
for (String permutation : permutations)
System.out.println(permutation);
} // main
//-------------------------------------------------------------------------------------
//------ Solution 1: For Input with Unique Characters Only -----------
/**
* Computes permutation of given string without any duplicate characters.
* For string duplicate character, results will contain duplicate entries.
* @param s input string
* @return List of permutations
*/
public static List<String> permutationWithoutDuplicates(String s) {
char[] chars = s.toCharArray();
List<String> permutations = new ArrayList<>();
permutationWithoutDuplicatesHelper(chars, 0, permutations);
return permutations;
} // permutationWithoutDuplicates
//-------------------------------------------------------------------------------------
/**
* Recursive helper method to compute permutations of a character array without duplicates.
* @param chars input character array
* @param cursor pointer to current location considered. Characters before this pointer and kept fixed.
* @param permutations List holding resulting permutations
*/
private static void permutationWithoutDuplicatesHelper(char[] chars, int cursor, List<String> permutations) {
if (cursor == chars.length) {
permutations.add(new String(chars));
return;
}
for (int i = cursor; i < chars.length; i++) {
swap(chars, cursor, i);
// keep the selected character fixed and compute permutations of remaining characters
permutationWithoutDuplicatesHelper(chars, cursor + 1, permutations);
swap(chars, cursor, i);
}
} // permutationWithoutDuplicatesHelper
//-------------------------------------------------------------------------------------
/**
* Utility method to swap two characters in the given character array.
* @param chars input character array
* @param i first index to swap.
* @param j second index to swap.
*/
private static void swap(char[] chars, int i, int j) {
if (i == j) return;
char temp = chars[i];
chars[i] = chars[j];
chars[j] = temp;
} // swap
//------ END Solution 1 ---------------------------------------------------------------
//-------------------------------------------------------------------------------------
//------ Solution 2: For Input with Duplicate Characters ------------------------------
/**
* Computes permutation of an string with Duplicate characters.
* Obviously, it will also work for the string with unique characters.
* @param s input string.
* @return Permutations of given input.
*/
public static List<String> permuteWithDuplicates(String s) {
char[] chars = s.toCharArray();
Map<Character, Integer> frequencyMap = createFrequencyMap(chars);
List<String> results = new ArrayList<>();
permuteWithDuplicatesHelper(frequencyMap, new char[s.length()], 0, results);
return results;
} // permuteWithDuplicates
//-------------------------------------------------------------------------------------
/**
* Recursive helper method to compute permutations of a string with duplicate characters.
* @param frequencyMap Map holding character counts of input.
* @param permutation current permutation
* @param cursor current location in permutation
* @param results List of all permutations of input.
*/
private static void permuteWithDuplicatesHelper(Map<Character, Integer> frequencyMap, char[] permutation, int cursor, List<String> results) {
if (cursor == permutation.length) {
results.add(new String(permutation));
return;
}
for (Map.Entry<Character, Integer> entry : frequencyMap.entrySet()) {
if (entry.getValue() > 0) {
permutation[cursor] = entry.getKey();
entry.setValue(entry.getValue() - 1);
permuteWithDuplicatesHelper(frequencyMap, permutation, cursor + 1, results);
entry.setValue(entry.getValue() + 1);
}
}
} // permuteWithDuplicatesHelper
//-------------------------------------------------------------------------------------
/**
* Utility method to create character counts as a Map.
* @param chars input character array
* @return Character Counts Map.
*/
private static Map<Character, Integer> createFrequencyMap(char[] chars) {
Map<Character, Integer> result = new HashMap<>();
for (Character c : chars) {
if (result.containsKey(c)) {
result.put(c, result.get(c) + 1);
} else {
result.put(c, 1);
}
}
return result;
} // createFrequencyMap
//------ END Solution 2 ---------------------------------------------------------------
} // Permutations

Time Complexity

There are n! permutations. We need to compute each of these permutations. The complexity of this approach is θ(n!) which is exponential with the size of the input.






Extract Text From PPTX using POI - java.lang.ArrayStoreException

I had been trying to extract textual content from PPTX file using Apache POI. Then I struggled with following exception.
 java.lang.ArrayStoreException 
  at java.lang.System.arraycopy(Native Method) 
  at org.apache.xmlbeans.impl.values.XmlObjectBase._typedArray(XmlObjectBase.java:409) 
  at org.apache.xmlbeans.impl.values.XmlObjectBase.selectPath(XmlObjectBase.java:457) 
  at org.apache.xmlbeans.impl.values.XmlObjectBase.selectPath(XmlObjectBase.java:415) 
  at org.apache.poi.xslf.usermodel.XSLFDrawing.<init>(XSLFDrawing.java:44) 
  at org.apache.poi.xslf.usermodel.XSLFSheet.initDrawingAndShapes(XSLFSheet.java:170) 
  at org.apache.poi.xslf.usermodel.XSLFSheet.getShapes(XSLFSheet.java:157) 
      

Solution

Well the problem turns out to be version of library I had been using.
I was running xmlbeans 2.3. Upgrading to 2.6 resolves the issue.