无重复字符串的排列组合。编写一种方法,计算某字符串的所有排列组合,字符串每个字符均不相同。
**示例1:
输入:S = “qwe”
输出:[“qwe”, “qew”, “wqe”, “weq”, “ewq”, “eqw”]
示例2:
输入:S = “ab”
输出:[“ab”, “ba”]
首先我想到的是把字符串变成char数组,然后好几个for循环,每一次循环拿走其中的一个。
但是后面我发现有以下几个问题:
- for循环每次拿走一个下一次循环怎么取回拿走的那个
- 当中嵌套的循环怎么确定要嵌套几次。
想不到这样该如何解决。就看了看其他人的解法,发现大多数人对于这种给定数列返回全部排列顺序的都是使用回溯法。
那么什么是回溯法呢?
我的理解就是例如不断向下寻找符合要求的子树,当寻找到符合条件的子树时,就向上返回。
回溯法的实现方法有两种:递归和递推(也称迭代)。一般来说,一个问题两种方法都可以实现,只是在算法效率和设计复杂度上有区别。
【类比于图深度遍历的递归实现和非递归(递推)实现】
1. 递归
思路简单,**设计容易,但效率低,**其设计范式如下:
[cpp] view plain copy
//针对N叉树的递归回溯方法
void backtrack (int t)
{
if (t>n) output(x); //叶子节点,输出结果,x是可行解
else
for i = 1 to k//当前节点的所有子节点
{
x[t]=value(i); //每个子节点的值赋值给x
//满足约束条件和限界条件
if (constraint(t)&&bound(t))
backtrack(t+1); //递归下一层
}
}
2. 递推
算法设计相对复杂,但效率高。
[cpp] view plain copy
//针对N叉树的迭代回溯方法
void iterativeBacktrack ()
{
int t=1;
while (t>0) {
if(ExistSubNode(t)) //当前节点的存在子节点
{
for i = 1 to k //遍历当前节点的所有子节点
{
x[t]=value(i);//每个子节点的值赋值给x
if (constraint(t)&&bound(t))//满足约束条件和限界条件
{
//solution表示在节点t处得到了一个解