递归
属于分治思想。
递归在于反复调用自身函数,每次把问题缩小为子问题,直到缩小到可以直接得到边界数据的结果,然后再往上返回,并在途中求出对应的解。
递归的逻辑要求:
①递归边界,即问题分解的尽头;
②递归式,将原问题分解成若干子问题的某种手段。
全排列
1~n这些整数所能形成的全部排列,有时需要实现按字典序从小到大排列。
问题:输出1~n这n个数字的全排列
分析:按照分治、递归的思想,该问题可以分为若干个子问题,即”输出以1开头的全排列“,”输出以2开头的全排列“等等…
①我们可以设定一个数组p用来存放当前的排列,其中每一个元素为一位数字;
②设定一个哈希表hash_table,用来标识1~n中的某一整数x已经存在数组p中,即已经用于排列。
解法:
按顺序往p的1~n位里面填数字。
假设p[1]~p[index-1]
都已经填好了,准备填p[index],如果此时x没有填入p[1]~p[index-1]
,即对应hash_table[x]=false
,那么把x填入p[index]
,同时要记录此时的hash_table[x]=true
。
然后开始处理p的第index+1
位,即开始递归。当递归完成后,将hash_table[x]
还原为false
。
确定递归边界,当index=n+1
时说明p的1~n位都已经填好,表示生成了一个排列。
完整代码如下:
#include <iostream>
using namespace std;
const int maxn = 11;
// n表示数字个数
// p表示当前排列
// hash_table用于记录整数x是否已经在p中
int n, p[maxn], hash_table[maxn] = {false};
//从排列的index位开始
void permutate(int index) {
if (index == n + 1) { //递归边界,已经处理完排列的1~n位
for (int i = 1; i <= n; i++) {
printf("%d", p[i]); //输出当前排列
}
printf("\n");
return;
}
for (int x = 1; x <= n; x++) { //枚举1~n,将x填入p中
//当前数字x没有被填入到p[0]~p[index-1]中
if (hash_table[x] == false) {
p[index] = x; //将x放入p[index]
hash_table[x] = true; //记录x已经放入了排列
permutate(index + 1); //递归处理index+1位
hash_table[x] = false; //处理完p[index]=x的子问题
}
}
}
int main() {
n = 3; //以3为例
permutate(1); //从排列的第一位开始
return 0;
}