原题如下:用1、2、2、3、4、5这六个数字,用java写一个main函数,打印出所有不同的排列,如:512234、412345等,要求: "4 "不能在第三位, "3 "与 "5 "不能相连.
我看了回贴都没有很好解决,主要是没有排除重复。
解决思路:强化题目,用1、2、2、3、4、5这六个数字排列“递增”序列。其他要求不变。
算法思路:显然是递归,初始序列122345,先从末两位(45)变化(45,54),然后末三位(345) ... 直到最后六位.怎样解决重复问题?很简单,由于是递增序列,每生成新序列可与前一生成序列比较,如 <放弃当前序列。当然有更好效率,如预先预测。
方法一代码如下:
class test
{
// 当前固定部分
private String CurFixPart;
private String PreGenNum;
public static void main(String[] args)
{
test t=new test();
t.GenControll( "122345 ");
}
// 调整字符串s位置pos字符到最前
private String shift(String s, int pos)
{
String newStr;
if (s.length()> pos+1)
newStr=s.substring(pos, pos+1)
+s.substring(0, pos)
+s.substring(pos+1);
else
newStr=s.substring(pos)
+s.substring(0, pos);
return newStr;
}
protected int Validate(String newNum)
{
String newGenNum=CurFixPart+newNum;
if (Integer.valueOf(newGenNum) <=Integer.valueOf(PreGenNum))
return 0;
if (newGenNum.substring(2,3).equals( "4 ") ||
(newGenNum.indexOf( "35 ")!=-1) || (newGenNum.indexOf( "53 ")!=-1))
return 0;
PreGenNum=newGenNum;
System.out.println(newGenNum);
return 0;
}
public void GenControll(String Base)
{
PreGenNum= "0 ";
CurFixPart= " ";
GenNext(Base, 0);
}
void GenNext(String varPart, int curPos)
{
if (varPart.length()==2)
{
Validate(varPart);
Validate(shift(varPart, 1));
return;
}
// Next Layer
String newGen=shift(varPart, curPos);
String SavedFixPart=CurFixPart;
CurFixPart=CurFixPart+newGen.substring(0,1);
GenNext(newGen.substring(1), 0);
CurFixPart=SavedFixPart;
// 同层递增
if (curPos==varPart.length()-1)
return;
GenNext(varPart, curPos+1);
}
}
序列122345测试通过。
方法二
其中输入的数组长度和重复的数字可以调整。但是规则是固定的。以后再考虑怎么完善成可以处理任意的字符串,规则也可以考虑用一个可配置的数组来表示。
大体的思路就是先对原始数组排序。并生成重复数字的标记数组。
在主算法的每一层循环对原始数组中的某一位数字的所有可能的位置做全枚举,同时当该位数字不是第一次出现的时候,其位置只能是上一位数字(重复数字)右侧。然后递归调用。
用 122345 和 1222345 测试 OK
import java.util.ArrayList;
import java.util.Arrays;
/**
* @author Owen Luo
* @version Created at 2008-11-6 下午12:51:48 用 ArrayList 和 StringBuffer 来实现较少的资源占用。
*/
public class CSDN_ArrayQuest_081106 {
public static void main(String[] args) {
// TODO Auto-generated method stub
long beginTime = System.currentTimeMillis();
CSDN_ArrayQuest_081106 newProcess = new CSDN_ArrayQuest_081106();
newProcess.preJobs();
for (int a = 0; a < newProcess.arrayLength; a++) {
newProcess.process(newProcess.posAL, a, 0);
}
long endTime = System.currentTimeMillis();
System.out.println(newProcess.counter);
System.out.println( "Computation time: "+ (endTime-beginTime));
}
// pozAL 剩下的位置列表,从小到大,顺序排列
// pozIndex 位置列表中遍历的起始位置
// digitIndex 目前正在遍历处理的数字在原始排序数组中的位置
public void process(ArrayList pozAL, int pozIndex, int digitIndex) {
// make a copy of pozAL
ArrayList tempAL = new ArrayList(pozAL.size());
for (int i = 0; i < pozAL.size(); i++) {
tempAL.add(Integer.valueOf(pozAL.get(i).intValue()));
}
posArray[digitIndex] = tempAL.get(pozIndex).intValue();
tempAL.remove(pozIndex);
int nextPozIndex;
if (digitIndex != arrayLength - 1) {
if (flagArray[digitIndex + 1] == 1) {
nextPozIndex = 0;
} else {
nextPozIndex = pozIndex; //如果是重复对象,只放右边
}
for (int i = nextPozIndex; i < tempAL.size(); i++) {
process(tempAL, i, digitIndex + 1);
}
} else {
genResult(posArray); //这里用一个 StringBuffer 对象,避免String类型的变量占用过多资源。
// 条件筛出 “35”,“53”,“4”在第二位的情况
if (result.indexOf("35") == -1 & result.indexOf("53") == -1
& result.indexOf("4") != 2) {
counter++;
System.out.println(result.toString());
}
}
}
public void preJobs() {
// sorting original number array
Arrays.sort(numArray);
// initialize posArray
posArray = new int[arrayLength];
for (int i = 0; i < arrayLength; i++) {
posArray[i] = i;
}
posAL = new ArrayList(arrayLength);
for (int i = 0; i < arrayLength; i++) {
posAL.add(Integer.valueOf(i));
}
// initialze flagArray
flagArray = new int[arrayLength];
for (int i = 0; i < arrayLength; i++) {
if (i == 0) {
flagArray[i] = 1;
} else {
flagArray[i] = (numArray[i - 1] == numArray[i]) ? 0 : 1;
}
}
}
public void genResult(int[] pozIndex) {
result.ensureCapacity(arrayLength);
for (int i = 0; i < arrayLength; i++) {
result.insert(pozIndex[i],numArray[i]);
result.delete(pozIndex[i]+1,pozIndex[i]+2);
}
}
int[] numArray = { 1, 2, 2, 2 ,3, 4, 5 };
int[] posArray;
int[] flagArray;
int arrayLength = numArray.length;
StringBuffer result = new StringBuffer("");
int counter = 0;
int counterALL = 0;
ArrayList posAL;
}
方法三
对于任意一个串利用递归进行排列时,我们是循环串中的每个字符到第一个字符进行递归。如果串中字符出现重复的话,则重复的字符只可以利用递归算法一次,即只要与前面相同的字符循环到第一个字符时不调用递归就可以避免重复,为此,我们只需要按如下方式修改算法:
import java.util.*;
public class test {
static int count=0;
public static void main(String[] arg) {
Scanner r=new Scanner(System.in);
String s=r.nextLine();
Pailie(s, " ");
System.out.println( "Total: "+count);
}
static void Pailie(String s,String p) {
if(s.length() <1) {
System.out.println(p);//字符串长度小于1,换行
count++;
}
else {
int index[]=new int[s.length()];
for(int i=0; i