生成给定字符串的所有置换

xdyibdwo  于 2021-09-13  发布在  Java
关注(0)|答案(30)|浏览(467)

找到字符串的所有排列的优雅方法是什么。e、 g.用于 ba ,将是 baab ,但对于较长的字符串,例如 defgh ? 是否有任何java实现示例?

bsxbgnwa

bsxbgnwa1#

如果有人想生成置换来处理它们,而不是通过void方法打印它们:

  1. static List<int[]> permutations(int n) {
  2. class Perm {
  3. private final List<int[]> permutations = new ArrayList<>();
  4. private void perm(int[] array, int step) {
  5. if (step == 1) permutations.add(array.clone());
  6. else for (int i = 0; i < step; i++) {
  7. perm(array, step - 1);
  8. int j = (step % 2 == 0) ? i : 0;
  9. swap(array, step - 1, j);
  10. }
  11. }
  12. private void swap(int[] array, int i, int j) {
  13. int buffer = array[i];
  14. array[i] = array[j];
  15. array[j] = buffer;
  16. }
  17. }
  18. int[] nVector = new int[n];
  19. for (int i = 0; i < n; i++) nVector [i] = i;
  20. Perm perm = new Perm();
  21. perm.perm(nVector, n);
  22. return perm.permutations;
  23. }
展开查看全部
x3naxklr

x3naxklr2#

使用es6的字符串置换
使用reduce()方法

  1. const permutations = str => {
  2. if (str.length <= 2)
  3. return str.length === 2 ? [str, str[1] + str[0]] : [str];
  4. return str
  5. .split('')
  6. .reduce(
  7. (acc, letter, index) =>
  8. acc.concat(permutations(str.slice(0, index) + str.slice(index + 1)).map(val => letter + val)),
  9. []
  10. );
  11. };
  12. console.log(permutations('STR'));
nhn9ugyo

nhn9ugyo3#

打印给定字符串的所有排列(考虑重复字符)并仅打印唯一字符的java实现如下所示:

  1. import java.util.Set;
  2. import java.util.HashSet;
  3. public class PrintAllPermutations2
  4. {
  5. public static void main(String[] args)
  6. {
  7. String str = "AAC";
  8. PrintAllPermutations2 permutation = new PrintAllPermutations2();
  9. Set<String> uniqueStrings = new HashSet<>();
  10. permutation.permute("", str, uniqueStrings);
  11. }
  12. void permute(String prefixString, String s, Set<String> set)
  13. {
  14. int n = s.length();
  15. if(n == 0)
  16. {
  17. if(!set.contains(prefixString))
  18. {
  19. System.out.println(prefixString);
  20. set.add(prefixString);
  21. }
  22. }
  23. else
  24. {
  25. for(int i=0; i<n; i++)
  26. {
  27. permute(prefixString + s.charAt(i), s.substring(0,i) + s.substring(i+1,n), set);
  28. }
  29. }
  30. }
  31. }
展开查看全部
d7v8vwbk

d7v8vwbk4#

下面是另一种更简单的字符串排列方法。

  1. public class Solution4 {
  2. public static void main(String[] args) {
  3. String a = "Protijayi";
  4. per(a, 0);
  5. }
  6. static void per(String a , int start ) {
  7. //bse case;
  8. if(a.length() == start) {System.out.println(a);}
  9. char[] ca = a.toCharArray();
  10. //swap
  11. for (int i = start; i < ca.length; i++) {
  12. char t = ca[i];
  13. ca[i] = ca[start];
  14. ca[start] = t;
  15. per(new String(ca),start+1);
  16. }
  17. }//per
  18. }
展开查看全部
7tofc5zh

7tofc5zh5#

  1. import java.io.IOException;
  2. import java.util.ArrayList;
  3. import java.util.Scanner;
  4. public class hello {
  5. public static void main(String[] args) throws IOException {
  6. hello h = new hello();
  7. h.printcomp();
  8. }
  9. int fact=1;
  10. public void factrec(int a,int k){
  11. if(a>=k)
  12. {fact=fact*k;
  13. k++;
  14. factrec(a,k);
  15. }
  16. else
  17. {System.out.println("The string will have "+fact+" permutations");
  18. }
  19. }
  20. public void printcomp(){
  21. String str;
  22. int k;
  23. Scanner in = new Scanner(System.in);
  24. System.out.println("enter the string whose permutations has to b found");
  25. str=in.next();
  26. k=str.length();
  27. factrec(k,1);
  28. String[] arr =new String[fact];
  29. char[] array = str.toCharArray();
  30. while(p<fact)
  31. printcomprec(k,array,arr);
  32. // if incase u need array containing all the permutation use this
  33. //for(int d=0;d<fact;d++)
  34. //System.out.println(arr[d]);
  35. }
  36. int y=1;
  37. int p = 0;
  38. int g=1;
  39. int z = 0;
  40. public void printcomprec(int k,char array[],String arr[]){
  41. for (int l = 0; l < k; l++) {
  42. for (int b=0;b<k-1;b++){
  43. for (int i=1; i<k-g; i++) {
  44. char temp;
  45. String stri = "";
  46. temp = array[i];
  47. array[i] = array[i + g];
  48. array[i + g] = temp;
  49. for (int j = 0; j < k; j++)
  50. stri += array[j];
  51. arr[z] = stri;
  52. System.out.println(arr[z] + " " + p++);
  53. z++;
  54. }
  55. }
  56. char temp;
  57. temp=array[0];
  58. array[0]=array[y];
  59. array[y]=temp;
  60. if (y >= k-1)
  61. y=y-(k-1);
  62. else
  63. y++;
  64. }
  65. if (g >= k-1)
  66. g=1;
  67. else
  68. g++;
  69. }
  70. }
展开查看全部
x33g5p2x

x33g5p2x6#

字符串的排列:

  1. public static void main(String args[]) {
  2. permu(0,"ABCD");
  3. }
  4. static void permu(int fixed,String s) {
  5. char[] chr=s.toCharArray();
  6. if(fixed==s.length())
  7. System.out.println(s);
  8. for(int i=fixed;i<s.length();i++) {
  9. char c=chr[i];
  10. chr[i]=chr[fixed];
  11. chr[fixed]=c;
  12. permu(fixed+1,new String(chr));
  13. }
  14. }
s5a0g9ez

s5a0g9ez7#

我的实现基于mark byers的上述描述:

  1. static Set<String> permutations(String str){
  2. if (str.isEmpty()){
  3. return Collections.singleton(str);
  4. }else{
  5. Set <String> set = new HashSet<>();
  6. for (int i=0; i<str.length(); i++)
  7. for (String s : permutations(str.substring(0, i) + str.substring(i+1)))
  8. set.add(str.charAt(i) + s);
  9. return set;
  10. }
  11. }
xwbd5t1u

xwbd5t1u8#

递归是没有必要的,即使你可以直接计算任何排列,这个解决方案使用泛型来排列任何数组。
这里有一个关于这个算法的好信息。
对于c#开发人员来说,这里是更有用的实现。

  1. public static void main(String[] args) {
  2. String word = "12345";
  3. Character[] array = ArrayUtils.toObject(word.toCharArray());
  4. long[] factorials = Permutation.getFactorials(array.length + 1);
  5. for (long i = 0; i < factorials[array.length]; i++) {
  6. Character[] permutation = Permutation.<Character>getPermutation(i, array, factorials);
  7. printPermutation(permutation);
  8. }
  9. }
  10. private static void printPermutation(Character[] permutation) {
  11. for (int i = 0; i < permutation.length; i++) {
  12. System.out.print(permutation[i]);
  13. }
  14. System.out.println();
  15. }

该算法具有o(n)时间和空间复杂度来计算每个置换。

  1. public class Permutation {
  2. public static <T> T[] getPermutation(long permutationNumber, T[] array, long[] factorials) {
  3. int[] sequence = generateSequence(permutationNumber, array.length - 1, factorials);
  4. T[] permutation = generatePermutation(array, sequence);
  5. return permutation;
  6. }
  7. public static <T> T[] generatePermutation(T[] array, int[] sequence) {
  8. T[] clone = array.clone();
  9. for (int i = 0; i < clone.length - 1; i++) {
  10. swap(clone, i, i + sequence[i]);
  11. }
  12. return clone;
  13. }
  14. private static int[] generateSequence(long permutationNumber, int size, long[] factorials) {
  15. int[] sequence = new int[size];
  16. for (int j = 0; j < sequence.length; j++) {
  17. long factorial = factorials[sequence.length - j];
  18. sequence[j] = (int) (permutationNumber / factorial);
  19. permutationNumber = (int) (permutationNumber % factorial);
  20. }
  21. return sequence;
  22. }
  23. private static <T> void swap(T[] array, int i, int j) {
  24. T t = array[i];
  25. array[i] = array[j];
  26. array[j] = t;
  27. }
  28. public static long[] getFactorials(int length) {
  29. long[] factorials = new long[length];
  30. long factor = 1;
  31. for (int i = 0; i < length; i++) {
  32. factor *= i <= 1 ? 1 : i;
  33. factorials[i] = factor;
  34. }
  35. return factorials;
  36. }
  37. }
展开查看全部
bq9c1y66

bq9c1y669#

另一种简单的方法是循环字符串,选择尚未使用的字符并将其放入缓冲区,继续循环,直到缓冲区大小等于字符串长度。我更喜欢这种回溯解决方案,因为:
易懂
易于避免重复
输出被排序
以下是java代码:

  1. List<String> permute(String str) {
  2. if (str == null) {
  3. return null;
  4. }
  5. char[] chars = str.toCharArray();
  6. boolean[] used = new boolean[chars.length];
  7. List<String> res = new ArrayList<String>();
  8. StringBuilder sb = new StringBuilder();
  9. Arrays.sort(chars);
  10. helper(chars, used, sb, res);
  11. return res;
  12. }
  13. void helper(char[] chars, boolean[] used, StringBuilder sb, List<String> res) {
  14. if (sb.length() == chars.length) {
  15. res.add(sb.toString());
  16. return;
  17. }
  18. for (int i = 0; i < chars.length; i++) {
  19. // avoid duplicates
  20. if (i > 0 && chars[i] == chars[i - 1] && !used[i - 1]) {
  21. continue;
  22. }
  23. // pick the character that has not used yet
  24. if (!used[i]) {
  25. used[i] = true;
  26. sb.append(chars[i]);
  27. helper(chars, used, sb, res);
  28. // back tracking
  29. sb.deleteCharAt(sb.length() - 1);
  30. used[i] = false;
  31. }
  32. }
  33. }

输入str:1231
输出列表:{1123、1132、1213、1231、1312、1321、2113、2131、2311、3112、3121、3211}
请注意,输出已排序,并且没有重复的结果。

展开查看全部
64jmpszr

64jmpszr10#

  1. //Rotate and create words beginning with all letter possible and push to stack 1
  2. //Read from stack1 and for each word create words with other letters at the next location by rotation and so on
  3. /* eg : man
  4. 1. push1 - man, anm, nma
  5. 2. pop1 - nma , push2 - nam,nma
  6. pop1 - anm , push2 - amn,anm
  7. pop1 - man , push2 - mna,man
  8. * /
  9. public class StringPermute {
  10. static String str;
  11. static String word;
  12. static int top1 = -1;
  13. static int top2 = -1;
  14. static String[] stringArray1;
  15. static String[] stringArray2;
  16. static int strlength = 0;
  17. public static void main(String[] args) throws IOException {
  18. System.out.println("Enter String : ");
  19. InputStreamReader isr = new InputStreamReader(System.in);
  20. BufferedReader bfr = new BufferedReader(isr);
  21. str = bfr.readLine();
  22. word = str;
  23. strlength = str.length();
  24. int n = 1;
  25. for (int i = 1; i <= strlength; i++) {
  26. n = n * i;
  27. }
  28. stringArray1 = new String[n];
  29. stringArray2 = new String[n];
  30. push(word, 1);
  31. doPermute();
  32. display();
  33. }
  34. public static void push(String word, int x) {
  35. if (x == 1)
  36. stringArray1[++top1] = word;
  37. else
  38. stringArray2[++top2] = word;
  39. }
  40. public static String pop(int x) {
  41. if (x == 1)
  42. return stringArray1[top1--];
  43. else
  44. return stringArray2[top2--];
  45. }
  46. public static void doPermute() {
  47. for (int j = strlength; j >= 2; j--)
  48. popper(j);
  49. }
  50. public static void popper(int length) {
  51. // pop from stack1 , rotate each word n times and push to stack 2
  52. if (top1 > -1) {
  53. while (top1 > -1) {
  54. word = pop(1);
  55. for (int j = 0; j < length; j++) {
  56. rotate(length);
  57. push(word, 2);
  58. }
  59. }
  60. }
  61. // pop from stack2 , rotate each word n times w.r.t position and push to
  62. // stack 1
  63. else {
  64. while (top2 > -1) {
  65. word = pop(2);
  66. for (int j = 0; j < length; j++) {
  67. rotate(length);
  68. push(word, 1);
  69. }
  70. }
  71. }
  72. }
  73. public static void rotate(int position) {
  74. char[] charstring = new char[100];
  75. for (int j = 0; j < word.length(); j++)
  76. charstring[j] = word.charAt(j);
  77. int startpos = strlength - position;
  78. char temp = charstring[startpos];
  79. for (int i = startpos; i < strlength - 1; i++) {
  80. charstring[i] = charstring[i + 1];
  81. }
  82. charstring[strlength - 1] = temp;
  83. word = new String(charstring).trim();
  84. }
  85. public static void display() {
  86. int top;
  87. if (top1 > -1) {
  88. while (top1 > -1)
  89. System.out.println(stringArray1[top1--]);
  90. } else {
  91. while (top2 > -1)
  92. System.out.println(stringArray2[top2--]);
  93. }
  94. }
  95. }
展开查看全部
2w2cym1i

2w2cym1i11#

//将每个字符插入arraylist

  1. static ArrayList al = new ArrayList();
  2. private static void findPermutation (String str){
  3. for (int k = 0; k < str.length(); k++) {
  4. addOneChar(str.charAt(k));
  5. }
  6. }
  7. //insert one char into ArrayList
  8. private static void addOneChar(char ch){
  9. String lastPerStr;
  10. String tempStr;
  11. ArrayList locAl = new ArrayList();
  12. for (int i = 0; i < al.size(); i ++ ){
  13. lastPerStr = al.get(i).toString();
  14. //System.out.println("lastPerStr: " + lastPerStr);
  15. for (int j = 0; j <= lastPerStr.length(); j++) {
  16. tempStr = lastPerStr.substring(0,j) + ch +
  17. lastPerStr.substring(j, lastPerStr.length());
  18. locAl.add(tempStr);
  19. //System.out.println("tempStr: " + tempStr);
  20. }
  21. }
  22. if(al.isEmpty()){
  23. al.add(ch);
  24. } else {
  25. al.clear();
  26. al = locAl;
  27. }
  28. }
  29. private static void printArrayList(ArrayList al){
  30. for (int i = 0; i < al.size(); i++) {
  31. System.out.print(al.get(i) + " ");
  32. }
  33. }
展开查看全部
4ioopgfo

4ioopgfo12#

无递归的java实现

  1. public Set<String> permutate(String s){
  2. Queue<String> permutations = new LinkedList<String>();
  3. Set<String> v = new HashSet<String>();
  4. permutations.add(s);
  5. while(permutations.size()!=0){
  6. String str = permutations.poll();
  7. if(!v.contains(str)){
  8. v.add(str);
  9. for(int i = 0;i<str.length();i++){
  10. String c = String.valueOf(str.charAt(i));
  11. permutations.add(str.substring(i+1) + c + str.substring(0,i));
  12. }
  13. }
  14. }
  15. return v;
  16. }
展开查看全部
ctzwtxfj

ctzwtxfj13#

我们可以使用阶乘计算以特定字母开头的字符串数量。
示例:以输入为例 d . (3!) == 6 字符串将以字符串的每个字母开头 d .

  1. static public int facts(int x){
  2. int sum = 1;
  3. for (int i = 1; i < x; i++) {
  4. sum *= (i+1);
  5. }
  6. return sum;
  7. }
  8. public static void permutation(String str) {
  9. char[] str2 = str.toCharArray();
  10. int n = str2.length;
  11. int permutation = 0;
  12. if (n == 1) {
  13. System.out.println(str2[0]);
  14. } else if (n == 2) {
  15. System.out.println(str2[0] + "" + str2[1]);
  16. System.out.println(str2[1] + "" + str2[0]);
  17. } else {
  18. for (int i = 0; i < n; i++) {
  19. if (true) {
  20. char[] str3 = str.toCharArray();
  21. char temp = str3[i];
  22. str3[i] = str3[0];
  23. str3[0] = temp;
  24. str2 = str3;
  25. }
  26. for (int j = 1, count = 0; count < facts(n-1); j++, count++) {
  27. if (j != n-1) {
  28. char temp1 = str2[j+1];
  29. str2[j+1] = str2[j];
  30. str2[j] = temp1;
  31. } else {
  32. char temp1 = str2[n-1];
  33. str2[n-1] = str2[1];
  34. str2[1] = temp1;
  35. j = 1;
  36. } // end of else block
  37. permutation++;
  38. System.out.print("permutation " + permutation + " is -> ");
  39. for (int k = 0; k < n; k++) {
  40. System.out.print(str2[k]);
  41. } // end of loop k
  42. System.out.println();
  43. } // end of loop j
  44. } // end of loop i
  45. }
  46. }
展开查看全部
dojqjjoe

dojqjjoe14#

下面是一个简单的java最小递归解决方案:

  1. public static ArrayList<String> permutations(String s) {
  2. ArrayList<String> out = new ArrayList<String>();
  3. if (s.length() == 1) {
  4. out.add(s);
  5. return out;
  6. }
  7. char first = s.charAt(0);
  8. String rest = s.substring(1);
  9. for (String permutation : permutations(rest)) {
  10. out.addAll(insertAtAllPositions(first, permutation));
  11. }
  12. return out;
  13. }
  14. public static ArrayList<String> insertAtAllPositions(char ch, String s) {
  15. ArrayList<String> out = new ArrayList<String>();
  16. for (int i = 0; i <= s.length(); ++i) {
  17. String inserted = s.substring(0, i) + ch + s.substring(i);
  18. out.add(inserted);
  19. }
  20. return out;
  21. }
展开查看全部
mwyxok5s

mwyxok5s15#

  1. /**Returns an array list containing all
  2. * permutations of the characters in s. */
  3. public static ArrayList<String> permute(String s) {
  4. ArrayList<String> perms = new ArrayList<>();
  5. int slen = s.length();
  6. if (slen > 0) {
  7. // Add the first character from s to the perms array list.
  8. perms.add(Character.toString(s.charAt(0)));
  9. // Repeat for all additional characters in s.
  10. for (int i = 1; i < slen; ++i) {
  11. // Get the next character from s.
  12. char c = s.charAt(i);
  13. // For each of the strings currently in perms do the following:
  14. int size = perms.size();
  15. for (int j = 0; j < size; ++j) {
  16. // 1. remove the string
  17. String p = perms.remove(0);
  18. int plen = p.length();
  19. // 2. Add plen + 1 new strings to perms. Each new string
  20. // consists of the removed string with the character c
  21. // inserted into it at a unique location.
  22. for (int k = 0; k <= plen; ++k) {
  23. perms.add(p.substring(0, k) + c + p.substring(k));
  24. }
  25. }
  26. }
  27. }
  28. return perms;
  29. }
展开查看全部

相关问题