全面解析回溯法:算法框架与问题求解

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
简介:

目录

什么是回溯法?

回溯法的通用框架

利用回溯法解决问题

总结与探讨

附:《算法设计手册》第7章其余面试题解答

 

  摘了一段来自百度百科对回溯法思想的描述:

在包含问题的所有解的解空间树中,按照深度优先搜索的策略,从根结点出发深度探索解空间树。当探索到某一结点时,要先判断该结点是否包含问题的解,如果包含,就从该结点出发继续探索下去,如果该结点不包含问题的解,则逐层向其祖先结点回溯。(其实回溯法就是对隐式图的深度优先搜索算法)。 若用回溯法求问题的所有解时,要回溯到根,且根结点的所有可行的子树都要已被搜索遍才结束。 而若使用回溯法求任一个解时,只要搜索到问题的一个解就可以结束。

   可以把回溯法看成是递归调用的一种特殊形式。其实对于一个并非编程新手的人来说,从来没使用过回溯法来解决问题的情况是很少见的,不过往往是“对症下药”,针对特定的问题进行解答。这些天看了《算法设计手册》回溯法相关内容,觉得对回溯法抽象的很好。如果说算法是解决问题步骤的抽象,那么这个回溯法的框架就是对大量回溯法算法的抽象。本文将对这个回溯法框架进行分析,并且用它解决一系列的回溯法问题。文中的回溯法采用递归形式。

  在进一步的抽象之前,先来回顾一下DFS算法。对于一个无向图如下图左,它的从点1开始的DFS过程可能是下图右的情况,其中实线表示搜索时的路径,虚线表示返回时的路径:

 

  可以看出,在回溯法执行时,应当:保存当前步骤,如果是一个解就输出;维护状态,使搜索路径(含子路径)尽量不重复。必要时,应该对不可能为解的部分进行剪枝(pruning)。

  下面介绍回溯法的一般实现框架:

复制代码
bool finished = FALSE; /* 是否获得全部解? */
backtrack(int a[], int k, data input)
{
    int c[MAXCANDIDATES]; /*这次搜索的候选 */
    int ncandidates; /* 候选数目 */
    int i; /* counter */
    if (is_a_solution(a,k,input))
    process_solution(a,k,input);
    else {
        k = k+1;
        construct_candidates(a,k,input,c,&ncandidates);
        for (i=0; i<ncandidates; i++) {
            a[k] = c[i];
            make_move(a,k,input);
            backtrack(a,k,input);
            unmake_move(a,k,input);
            if (finished) return; /* 如果符合终止条件就提前退出 */
        }
    }
}
复制代码

  对于其中的函数和变量,解释如下:

  a[]表示当前获得的部分解;

  k表示搜索深度;

  input表示用于传递的更多的参数;

  is_a_solution(a,k,input)判断当前的部分解向量a[1...k]是否是一个符合条件的解

  construct_candidates(a,k,input,c,ncandidates)根据目前状态,构造这一步可能的选择,存入c[]数组,其长度存入ncandidates

  process_solution(a,k,input)对于符合条件的解进行处理,通常是输出、计数等

  make_move(a,k,input)unmake_move(a,k,input)前者将采取的选择更新到原始数据结构上,后者把这一行为撤销。

 

  其实回溯法框架就是这么简单,通过这个框架,足以解决很多回溯法问题了。不信?下面展示一下:

  (由于后文所有代码均为在C中编写,因此bool类型用int类型代替,其中0为FALSE,非0为TRUE。)

 

问题1:求一个集合的所有子集

解答:

  将3个主要的函数实现,这个问题就解决了。由于每次for循环中a[k]=c[i],这是唯一的改动,并且在下次循环时会被覆盖,不需要专门编写make_move()和make_unmove()。

复制代码
int is_a_solution(int a[],int k, data input)
{
    return k==input;
}

void construct_candidates(int a[],int k, data input, int c[],int *ncandidates) 
{
    c[0] = 1;
    c[1] = 0;
    *ncandidates = 2;
}

void process_solution(int a[],int k,data input)
{
    int i;
    printf("{");
    for(i=1;i<=k;i++)
        if(a[i])
            printf(" %d",i);
    printf(" }\n");
}
复制代码

  候选构造函数construct_candidates()相对简单,因为对每个集合中的元素和一个特定子集,只有出现和不出现这两种可能。

  调用这个函数只需:

generate_subsets(int n)
{
  int a[NMAX];
  backtrack(a,0,n);
}

扩展:

  Skiena在《算法设计手册》第14章组合算法部分介绍了生成子集的三种方式:按排序生成、二进制位变换、格雷码。上面这个算法是二进制变换的一种,格雷码生成可以参考后面习题解答的7-18;而按排序生成则比较复杂,它按特定顺序生成,如{1,2,3}生成顺序为{} , {1}, {1, 2}, {1, 2, 3}, {1, 3}, {2}, {2, 3},并且建议除非有这种要求,否则不要使用这个方式。 

 

 

问题2:输出不重复数字的全排列

解答:

  与上1题不同的是,由于不能重复出现,每次选择的元素都将影响之后的候选元素集合。构造候选时应从之前获得的部分解中获取信息,哪些元素是可以后续使用的,哪些是不可以的:

复制代码
void construct_candidates(int a[],int k, data input, int c[],int *ncandidates) 
{
    int i;
    int in_perm[NMAX+1];
    for(i=1;i<=NMAX;i++)
        in_perm[i] = 0;
    for(i=1;i<k;i++)
        in_perm[a[i]] = 1;
    *ncandidates = 0;
    for(i=1;i<=input;i++)
        if(!in_perm[i]) {
            c[*ncandidates] = i;
            *ncandidates += 1;
        }
}
复制代码

  不过这里可以看出一个问题,如果每次都是需要选择分支时构造候选元素,势必会造成浪费。这里仅仅是一个展示,如果提高效率,可以把解空间和原空间优化到一起,这样不必每次都生成解空间。下面的代码是对这个问题更好的也是更常见的解法,我相信不少人都写过,并对上一种看似复杂的解法表示不屑一顾:

复制代码
void permutaion(int *array,int k,int length)
{
    int i;
    if (length==k) {
        for(i=0;i<length;i++)
            printf("%d ",array[i]);
        printf("\n");
        return;
    }
                
    for(i=k;i<length;i++) {
        swap(&array[i],&array[k]);
        permutaion(array,k+1,length);
        swap(&array[i],&array[k]);
    }
}
复制代码

  但仔细观察这个解法,可以发现其实它暗含了is_a_solution()、construct candidates()、process_solution()、make_move()和unmake_move()这些步骤,它其实是一般的回溯法框架的简化和优化。

 

问题3:求解数独——剪枝的示范

解答:

  由于填入的数字涉及到横纵两个坐标,单纯的解向量a[]不能满足保存解的要求了。仅a[k]表示填入的值,定义一个结构以保存数独的当前状态和第k步时填入点的坐标:

复制代码
#define DIMENSION 9
#define NCELLS DIMENSION*DIMENSION
typedef struct {
    int x,y;
} point;

typedef struct {
    int m[DIMENSION+1][DIMENSION+1];
    int freecount;
    point move[NCELLS+1];
} boardtype;

typedef boardtype* data;
复制代码

  同时把获取下一步候选的construct_candidates()分解为两步:获取下一步填入点的坐标next_square()、获取该点可以填入的数值possible_values()。对于这两步需要进行一些探讨:

  next_square()可以采取任意取一个没有填入的点的随机策略(arbitrary);而更有效的策略是最大约束(Most Constrained),即取的点行、列以及所在3*3方阵点数最多的点,这样它的约束最多,填入的数字的可能性最少。

  possible_values()也可以采用两种策略:局部计数(local count),即只要满足行、列、3*3方阵内部都不冲突,就作为可能填入的数值;预测(look ahead),即对填入的数,预先找下一步时是否所有空都可填入至少一个数字来确认这个数是否可以被填入。《算法设计手册》作者认为,我们如果采用最大约束和局部计数策略,回溯过程就已经暗含了预测(失败时会回退),我曾经试过,专门写一个look ahead函数是得不偿失的,它并不比直接回溯开销小,甚至更大。

  因此,为了提高效率,next_square()采取最大约束策略,possible_values()采取暗含的预测策略。为了计算出最大约束的点,我还写了一个evaluate()函数用来计算某个未填点的得分,得分越大说明约束越强,约束最强的点将成为候选点。这个evaluate()不是很严格,因为它重复计算了一些点,不过影响不大。这两个策略的采取可以看作是剪枝的过程。剪枝是回溯法的重要加速途径,好的剪枝策略能够提高回溯法的运行速度,这是回溯法与暴力算法的一大区别。

void construct_candidates(int a[],int k,boardtype *board, int c[],int *ncandidates)
{
    int x,y;
    int i;
    int possible[DIMENSION+1];
    next_square(&x,&y,board);
    board->move[k].x = x;
    board->move[k].y = y;
    //printf("k:%d left:%d\n",k,board->freecount);
    *ncandidates = 0;
    if(x<0 && y<0)
        return;
    possible_values(x,y,board,possible);
    for(i=1;i<=DIMENSION;i++)
        if(possible[i]) {
            c[*ncandidates] = i;
            *ncandidates += 1;
        }
}

//most constrained square selection
void next_square(int *x,int *y, boardtype *board)
{
    int m_x,m_y,i,j;
    int score,max_score;
    m_x = -1,m_y = -1, max_score = 0;
    for(i=1;i<=DIMENSION;i++)
        for(j=1;j<=DIMENSION;j++) {
            if(board->m[i][j]) //not blank
                continue;
            score = evaluate(i,j,board);
            if(score > max_score) {
                m_x = i;
                m_y = j;
                max_score = score;
            }
        }
    *x = m_x;
    *y = m_y;
}

int evaluate(int x,int y,boardtype* board)
{
    int i,j,i_start,j_start;
    int score = 0;
    
    //row
    i = x;
    for(j=1;j<=DIMENSION;j++)
        score += (board->m[i][j] > 0);

    //column
    j=y;
    for(i=1;i<=DIMENSION;i++)
        score += (board->m[i][j]>0);

    //3*3 square
    i_start = (i-1)/3 *3 +1;
    j_start = (j-1)/3 *3 +1;
    //the most left and up point in the 3*3 square
    for(i=i_start;i<=i_start+2;i++)
        for(j=j_start;j<j_start+2;j++)
            score += (board->m[i][j]>0);
    return score;
}

int possible_values(int x,int y,boardtype* board, int possible[])
{
    int i,j;
    volatile int i_start,j_start;
    for(i=1;i<=DIMENSION;i++)
        possible[i] = 1;
    
    //row
    i = x;
    for(j=1;j<=DIMENSION;j++)
        possible[board->m[i][j]] = 0;

    //column
    j = y;
    for(i=1;i<=DIMENSION;i++)
        possible[board->m[i][j]] = 0;

    //3*3 square
    i_start = (x-1)/3;
    i_start = i_start * 3 + 1;
    j_start = (y-1)/3;
    j_start = j_start * 3 + 1; 
    //printf("i_start:%d j_start:%d\n",i_start,j_start);
    //the most left and up point in the 3*3 square
    for(i=i_start;i<=i_start+2;i++)
        for(j=j_start;j<=j_start+2;j++)
            possible[board->m[i][j]] = 0;
    //printf("(%d,%d):",x,y);
    //for(i=1;i<=DIMENSION;i++)
        //if(possible[i]) {
            //printf("%d ",i);
        //}
    return 0;
}
construct_candidates()、next_square()、possible_values()

  由于要对定义的数据结构进行修改,make_move()和unmake_move()也需要进行实现了。

void make_move(int a[], int k, boardtype *board)
{
    fill_square(board->move[k].x,board->move[k].y,a[k],board);
}

void unmake_move(int a[], int k, boardtype *board)
{
    free_square(board->move[k].x,board->move[k].y,board);
}

void fill_square(int x,int y,int key,boardtype* board){
    board->m[x][y] = key;
    board->freecount--;
}

void free_square(int x,int y,boardtype* board) {
    board->m[x][y] = 0;
    board->freecount++;
}
make_move()和unmake_move()

  is_a_solution()是对freecount是否为0的判断,process_solution()可以用作输出填好的数独,这两个函数的解法略过。而backtrack()函数和基本框架相比,看上去没多大的区别。

void backtrack(int a[],int k, boardtype* input)
{
    int c[DIMENSION];
    int ncandidates;
    int i;
    if(is_a_solution(a,k,input))
        process_solution(a,k,input);
    else {
        k = k+1;
        construct_candidates(a,k,input,c,&ncandidates);
        for(i=0;i<ncandidates;i++) {
            a[k] = c[i];
            make_move(a,k,input);
            backtrack(a,k,input);
            unmake_move(a,k,input);
            if (finished)
                return;
        }
    }
}
backtrack of sudoku

  经测试,《算法设计手册》上的Hard级别的数独,我的这个程序可以获得和原书同样的解。

附注:

  这里是以数独为例展示回溯法。而如果需要专门进行数独求解,可以试试DancingLinks,有一篇文章对其进行介绍,感兴趣的读者可以自行查阅。另外有关DancingLinks的性能,可以参阅:算法实践——舞蹈链(Dancing Links)算法求解数独

 

问题4:给定一个字符串,生成组成这个字符串的字母的全排列(《算法设计手册》面试题7-14)

解答:

  如果字符串内字母不重复,显然和问题2一样。

  如果字符串中有重复的字母,就比较麻烦了。不过套用回溯法框架仍然可以解决,为了简化候选元素的生成,将所有候选元素排列成数组,形成“元素-值”对,其中值代表这个元素还能出现几次,把ASCII码的A~Z、a~z映射为数组下标0~51。实现如下:

int is_a_solution(char a[],int k, int len) {
    return (k==len);
}

void process_solution(char a[],int k, int len) {
    int i;
    for(i=1;i<=k;i++)
        printf("%c",a[i]);
    printf("\n");
}

void backtrack(char a[],int k, int len,int candidate[])
{
    int i;
    if(is_a_solution(a,k,len))
        process_solution(a,k,len);
    else {
        k = k+1;
        for(i=0;i<MAXCANDIDATES;i++) {
            if(candidate[i]) {
                a[k] = i+ 'A' ;
                candidate[i] --;//make_move
                backtrack(a,k,len,candidate);
                candidate[i]++;//unmake_move
                if (finished)
                    return;
            }
        }
    }
}

void generate_permutations_of_string(char *p) 
{
    //sort
    char a[NMAX];
    int candidate[MAXCANDIDATES];
    int i,len=strlen(p);
    for(i=0;i<MAXCANDIDATES;i++)
        candidate[i] = 0;
    for(i=0;i<len;i++)
        candidate[p[i] - 'A']++;
    backtrack(a,0,len,candidate);
}
问题4解法

显然,construct_candidates()已经化入了backtrace()内部,而且这也是一个对如何将候选也作为参数传递给下一层递归的很好的展示

 

问题5:求一个n元集合的k元子集(n>=k>0)。(《算法设计手册》面试题7-15)

解答:

  如果想采用问题1的解法,需要稍作修改,使得遍历至叶结点(也即所有元素都进行标记是否在集合中)时,判断是不是一个解,即元素数目是否为k。满足才能输出。

#include <stdio.h>
#define MAXCANDIDATES 2
#define NMAX 3
typedef int data;

int is_a_solution(int a[],int k, data input);
void construct_candidates(int a[],int k,data input, int c[],int *ncandidates);
void process_solution(int a[],int k, data input);

static int finished = 0;

void construct_candidates(int a[],int k, data input, int c[],int *ncandidates) 
{
    c[0] = 1;
    c[1] = 0;
    *ncandidates = 2;
}

void process_solution(int a[],int k,data input)
{
    int i;
    printf("{");
    for(i=1;i<=k;i++)
        if(a[i])
            printf(" %d",i);
    printf(" }\n");
}

backtrack(int a[],int k, data input,int n,int num)
{
    int c[MAXCANDIDATES];
    int ncandidates;
    int i;
    if(n == num) {//is a solution
        process_solution(a,k,input);
        return;
    }
    else if ((num>n)||(k==input))//not a solution
        return;
    else{
        k=k+1;
        construct_candidates(a,k,input,c,&ncandidates);
        for(i=0;i<ncandidates;i++) {
            a[k] = c[i];
            if(c[i]) {
                num++;
                backtrack(a,k,input,n,num);
                num--;
            }
            else
                backtrack(a,k,input,n,num);

            if (finished)
                return;
        }
    }
}

generate_subsets(int n)
{
    int a[NMAX+1];
    backtrack(a,0,n,2,0);
}

int main()
{
    generate_subsets(NMAX);
}
问题5解法完整示例

 

问题6:电话号码对应字符串

  电话键盘上有9个数字,其中2~9分别代表了几个字母,如2:ABC,3:DEF......等等。给定一个数字序列,输出它所对应的所有字母序列。(《算法设计手册》面试题7-17,以及《编程之美》3.2“电话号码对应英语单词”)

解答:

  这个问题在回溯法里已经很简单了,因为每一步的选择都不影响下一步的选择。稍微要注意的一点是如何把数字与多个字母的对应关系告诉程序。这个存储结构和相应的construct_candidates()可能是这样的:

复制代码
static char ch[10][4] =
{
    "",
    "",
    "ABC",
    "DEF",
    "GHI",
    "JKL",
    "MNO",
    "PQRS",
    "TUV",
    "WXYZ",
};

static int total[10] ={0,0,3,3,3,3,3,4,3,4}; 

void construct_candidates(int a[],int k,data input, int *c,int *ncandidates)
{
    *c = input[k-1] - '0';
    *ncandidates = total[*c];
    return;
}
复制代码

  而backtrack()中填充解空间a[]则是这样的:

a[k] = ch[c][i];

  你会发现,backtrack()和《编程之美》3.2节解法二的RecurisiveSearch()实质是一样的:都是回溯法嘛。当然,能简化还是应该尽量简化的。

复制代码
//c[i][j] 数字i对应的第j个字母
//total[i] 数字i一共对应几个字母
//number[] 待转换的数字序列
//answer[] 解空间
//index 当前处理的数字的位置
//n 电话号码总长度

void recursive(int * number, int * answer, int index, int n)
{
    if(index == n)
    {
        for(int i=0;i<n;i++)
            printf("%c", c[number[i]][answer[i]]);
        printf("\n");
        return;
    }
    for(answer[index]=0;answer[index]<total[number[index]];answer[index]++)
    {
        recursive(number, answer, index+1, n);
    }
}
复制代码

 

问题7:一摞烙饼的排序(《编程之美》1.3)

  假设有一堆烙饼,大小不一,需要把它们摆成从下到上由大到小的顺序。每次翻转只能一次翻转最上面的几个烙饼,把它们上下颠倒。反复多次可以使烙饼有序。那么,最少应该翻转几次呢?

解答:

  根据《编程之美》的分析可知,对于n个烙饼,如果每次都把最大的先翻到最上面,然后再把它翻到最下面,这样就只用处理最上面的(n-1)个。而翻完n-1个时,最小的必然已经在上面,因此,翻转的上界是2(n-1)。

  为了在搜索解的时候剪枝,如果当前翻转次数多于上界2(n-1),则必然不是最少的,应该直接返回。

  同时,当烙饼内部几个部分分别有序时(比如3、4、5已经连在一起、9、10已经连在一起),不应该拆散它们,而是应该视为一个整体进行翻转。这样,每次把最大的和次大的翻在一起,肯定要优于上界。把这个不怎么紧密的下界记为LowerBound,值为顺序相邻两个烙饼大小不相邻顺序的对数(pairs,不是log)。

  这样,有了粗略的上界和下界,就可以进行剪枝了。为了更有效的剪枝,可以把当前翻转步数大于已记录解的翻转步数的所有解也给剪掉。

  套用回溯法的框架,以下是求解代码。虽然和《编程之美》上的C++的面向对象版本看上去不太一样,但实质是一样的:

#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
typedef int* data;

int is_a_solution(int a[],int k, int step);
//void construct_candidates(int a[],int k,data input, int c[],int *ncandidates);
void process_solution(int a[],int n,int step, data input,data output);
//void make_move(int a[],int k, data input);
//void unmake_move(int a[],int k,data inupt);

int LowerBound(int *CakeArray,int n);
int UpperBound(int n);
void Reverse(int CakeArray[],int begin,int end);
void generate_sort(int a[],int n);
void show(data output);
static int maxStep;

//is_sorted()
int is_a_solution(int *CakeArray,int n,int step)
{
    int i;
    for(i=1;i<n;i++)
        if(CakeArray[i-1]>CakeArray[i])
            return 0;
    if(step<maxStep)
        return 1;
    return 0;
}

void process_solution(int a[],int n,int step,data input,data output)
{
    int i;
    maxStep = step;
    printf("new maxStep:%d\n",maxStep);
    for(i=0;i<step;i++)
        output[i] = input[i];
    return;
}

int UpperBound(int n)
{
    return 2*(n-1);
}

int LowerBound(int *CakeArray,int n)
{
    int i,t,ret = 0;
    for(i=1;i<n;i++)
    {
        t = CakeArray[i-1] - CakeArray[i];
        if((t==1)||(t==-1))
            continue;
        else
            ret++;
    }
    return ret;
}

void Reverse(int CakeArray[],int begin,int end) {
    assert(end>begin);
    int i,j,t;
    for(i=begin,j=end;i<j;i++,j--)
    {
        t = CakeArray[i];
        CakeArray[i] = CakeArray[j];
        CakeArray[j] = t;
    }
}

backtrack(int a[],int n,int step, data input,data output)
{
    int i;
    if(step+LowerBound(a,n)>maxStep)
        return;
    if(is_a_solution(a,n,step))
        process_solution(a,n,step,input,output);
    else {
        //construct_candidates(a,k,input,c,&ncandidates);
        for(i=1;i<n;i++) {
            Reverse(a,0,i);//make_move(a,k,input);
            input[step] = i;
            backtrack(a,n,step+1,input,output);
            Reverse(a,0,i);//unmake_move(a,k,input);
        }
    }
}

void generate_sort(int a[],int n)
{
    maxStep = UpperBound(n);
    int *SwapArray = malloc((UpperBound(n)+1)*sizeof(int));
    int *minSwapArray = malloc((UpperBound(n)+1)*sizeof(int));
    backtrack(a,n,0,SwapArray,minSwapArray);
    show(minSwapArray);
}

void show(data output)
{
    int i;
    for(i=0;i<maxStep;i++)
        printf("%d",output[i]);
    printf("\n");
    return ;
}


int main() {
    int i,n;
    printf("number of pancake:");
    scanf("%d",&n);
    int *CakeArray = malloc(n*sizeof(int));
    printf("pancakes' order(continuously):\n");
    for(i=0;i<n;i++)
        scanf("%d",&CakeArray[i]);
    printf("searching...\n");
    generate_sort(CakeArray,n);
    return 0;
}
烙饼排序

  其实理解这个算法的关键是如何把“翻转烙饼”的过程抽象成数据结构的改变,回溯法倒不是那么重要。

 

问题8:8皇后问题

  国际象棋棋盘上有8*8个格子。现在有8枚皇后棋子,一个格子只能放一个棋子,求解所有放法,使得这些棋子不同行、不同列、且不在对角线上((4,5)和(5,6)就是在对角线上的情况,不合法)。

解答:

  上面练习了那么多回溯法的问题,我相信能看到这里的人水平已经足以解决这个问题了。按行放置可以保证棋子不同行,对于每种放置可能,检查是否与上面各行的棋子是否同列、同对角线。都不满足的才能选作此次的决策即可。

#include <stdio.h>
#define DIM 8

int is_a_solution(int a[DIM][DIM],int row);
//void construct_candidates(int a[],int k,data input, int c[],int *ncandidates);
void process_solution(int a[DIM][DIM]);
//void make_move(int a[],int k, data input);
//void unmake_move(int a[],int k,data inupt);

static int finished = 0;
static count = 0;

int is_a_solution(int chess[DIM][DIM],int row)
{
    return (row == DIM);
}

void process_solution(int chess[DIM][DIM])
{
    int i,j;
    count++;
    for(i=0;i<DIM;i++) {
        for(j=0;j<DIM;j++)
            printf("%d ",chess[i][j]);
        printf("\n");
    }
    printf("\n");
}

int is_collision(int chess[DIM][DIM],int x,int y)
{
    int i,j;
    for(i=x-1,j=y-1;i>=0 && j>=0;i--,j--)
        if(chess[i][j] == 1)
            return 1;

    for(i=x-1,j=y+1;i>=0 && j<DIM;i--,j++)
        if(chess[i][j] == 1)
            return 1;

    return 0;
}

backtrack(int chess[DIM][DIM],int row, int* candidates)
{
    if(is_a_solution(chess,row))
        process_solution(chess);
    else {
        int i;
        //construct_candidates(a,k,input,c,&ncandidates);
        for(i=0;i<DIM;i++) {
            if(candidates[i] || is_collision(chess,row,i))
                continue;
            //make_move(a,k,input);
            chess[row][i] = 1;
            candidates[i] = 1;

            backtrack(chess,row+1,candidates);

            //unmake_move(a,k,input);
            chess[row][i] = 0;
            candidates[i] = 0;
        }
    }
}

void generate_8queen(int chess[8][8])
{
    int candidates[DIM] = {0,0,0,0,0,0,0,0};
    backtrack(chess,0,candidates);
    printf("total:%d\n",count);
    return;
}

int main()
{
    int chess[8][8] = {
        {0,0,0,0,0,0,0,0},
        {0,0,0,0,0,0,0,0},
        {0,0,0,0,0,0,0,0},
        {0,0,0,0,0,0,0,0},
        {0,0,0,0,0,0,0,0},
        {0,0,0,0,0,0,0,0},
        {0,0,0,0,0,0,0,0},
        {0,0,0,0,0,0,0,0}
    };
    generate_8queen(chess);
}
8皇后问题

 

总结与探讨

  通过以上的实例,可以发现回溯法框架确实能够解决许多形态各异的问题,这也得归功于这个框架足够抽象而不限于具体问题的求解,其通用性毋庸置疑。

  然而如果一个问题看到之后就有了思路,并能直接写出类似于问题2的精简版的情况又如何呢?这种情况下当然就没必要再去套用回溯法框架了,因为你已经把这个框架的步骤内化到自己的思考中并能在这个问题上运用自如了,这一点是值得高兴的。这时回溯法框架对于你来说只是用于检查代码正确性的一种额外验证方式罢了,没必要退而求其次。

  当你思路比较混乱,不知如何下手时我才建议搬出回溯法框架进行分析和套用。不过从问题7烙饼排序中可以看到,有时思路的不清晰往往是对实际问题的抽象不够,而不是编写回溯法解决本身的问题。

  编写回溯法时应该注意尽可能剪枝,同时维护好构造候选时所用的数据结构。 

 

附:《算法设计手册》第7章其余面试题解答

7-16.

  请用给定字符串中的字母重新组合成在字典中的单词。比如Steven Skiena可以重组为Vainest Knees。

解答:

  虽然通过回溯法可以把所有情况列出并与字典对照,但这未免太没有效率了。

  更快的方法是把给定字符串和所有字典单词排序成字母序,比如apple变成aelpp,再对排序后的字符串在排序后的字典进行搜索。这是个变位词的变形,变位词的处理可以参考:http://www.cnblogs.com/wuyuegb2312/p/3139926.html#title21

 

7-18.

  一间能容纳n个人的空房,房外有n个人。你站在门口,可以选择让门外的一个人进屋,也可以选择让屋内的人出来一个。请输出所有的2n种屋中人的出现情况的可能,并且这些情况是相邻的(上一种情况通过一次操作能变成下一种情况)

解答:

  一开始不是很理解,参考答案上也提到是用格雷码来解决。不过如果知道格雷码的生成方式,就好解决了:

  1. 1位格雷码有两个码字
  2. (n+1)位格雷码中的前2 n个码字等于n位格雷码的码字,按顺序书写,加前缀0
  3. (n+1)位格雷码中的前2 n个码字等于n位格雷码的码字,按逆序书写,加前缀1

 

  不过为了输出美观,由于C语言不提供printf直接输出2进制数,需要把10进制数转化成2进制数输出,而且首端的0要补上,这需要花点心思。下面是一个生成4位格雷码的程序,并不是回溯法。(为了省事直接在回溯法框架上改的)

#include <stdio.h>
#include <stdlib.h>
#include <assert.h>

void process_solution(int a[],int k, int n);
int print_bin(int num);
int e2(int n);

void gray_code(int a[],int k, int n)
{
    int i,j;
    if(k==n+1)
        process_solution(a,k,n);
    else {
        for(i=e2(k)-1,j=e2(k);i>=0;i--,j++) 
            a[j] = (1<<k)+a[i];
        gray_code(a,k+1,n);
    }
}

void process_solution(int a[],int k,int n)
{
    int i;
    int j = e2(n);
    for(i=0;i<j;i++) {
        print_bin(a[i]);
        printf("\n");
    }
    return;
}

int print_bin(int num)
{
    int pes_num=0;
    int i=1,bits=4;
    if(num==0)
        bits--;
    while(num>0) {
        if(num%2)
            pes_num += i;
        i *= 10;
        num /= 2;
        bits--;
    }
    for(;bits>0;bits--)
        printf("0");
    printf("%d",pes_num);
    return 0;
}

int e2(int n)
{
    int res = 1;
    assert(n<16 && n>= 0);
    while(n>0) {
        res *= 2;
        n--;
    }
    return res;
}

int generate_gray_code(int n)
{
    int *a = malloc(e2(n)*sizeof(int));
    a[0] = 0;
    gray_code(a,0,4);
}

int main()
{
    generate_gray_code(4);
    //int i;
    //for(i=0;i<16;i++)
    //{
    //    print_bin(i);
    //    printf("\n");
    //}
}
格雷码生成

 

7-19.

  使用能生成随机数{0,1,2,3,4}的函数rng04来生成rng07。每次运行rng07平均要调用几次rng04?

解答:

  随机数生成函数以前已经分析过了:http://www.cnblogs.com/wuyuegb2312/p/3141292.html。对于调用次数的期望,

  如果将rng07写作rng03+4*rng01,那么rng04调用的次数为它在rng03和rng01之和,都是 $1\cdot \frac{4}{5} +2\cdot \frac{4}{5}\cdot \frac{1}{5} +3\cdot \frac{4}{5}\cdot (\frac{1}{5})^{2}+ ... = 1.25$ 

 

 







本文转自五岳博客园博客,原文链接:www.cnblogs.com/wuyuegb2312/p/3273337.html,如需转载请自行联系原作者

目录
相关文章
|
2天前
|
算法 搜索推荐 Java
【潜意识Java】深度解析黑马项目《苍穹外卖》与蓝桥杯算法的结合问题
本文探讨了如何将算法学习与实际项目相结合,以提升编程竞赛中的解题能力。通过《苍穹外卖》项目,介绍了订单配送路径规划(基于动态规划解决旅行商问题)和商品推荐系统(基于贪心算法)。这些实例不仅展示了算法在实际业务中的应用,还帮助读者更好地准备蓝桥杯等编程竞赛。结合具体代码实现和解析,文章详细说明了如何运用算法优化项目功能,提高解决问题的能力。
32 6
|
28天前
|
设计模式 XML Java
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能
本文详细介绍了Spring框架的核心功能,并通过手写自定义Spring框架的方式,深入理解了Spring的IOC(控制反转)和DI(依赖注入)功能,并且学会实际运用设计模式到真实开发中。
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能
|
21天前
|
存储 算法 安全
基于红黑树的局域网上网行为控制C++ 算法解析
在当今网络环境中,局域网上网行为控制对企业和学校至关重要。本文探讨了一种基于红黑树数据结构的高效算法,用于管理用户的上网行为,如IP地址、上网时长、访问网站类别和流量使用情况。通过红黑树的自平衡特性,确保了高效的查找、插入和删除操作。文中提供了C++代码示例,展示了如何实现该算法,并强调其在网络管理中的应用价值。
|
1月前
|
机器学习/深度学习 人工智能 算法
深入解析图神经网络:Graph Transformer的算法基础与工程实践
Graph Transformer是一种结合了Transformer自注意力机制与图神经网络(GNNs)特点的神经网络模型,专为处理图结构数据而设计。它通过改进的数据表示方法、自注意力机制、拉普拉斯位置编码、消息传递与聚合机制等核心技术,实现了对图中节点间关系信息的高效处理及长程依赖关系的捕捉,显著提升了图相关任务的性能。本文详细解析了Graph Transformer的技术原理、实现细节及应用场景,并通过图书推荐系统的实例,展示了其在实际问题解决中的强大能力。
230 30
|
25天前
|
存储 监控 算法
企业内网监控系统中基于哈希表的 C# 算法解析
在企业内网监控系统中,哈希表作为一种高效的数据结构,能够快速处理大量网络连接和用户操作记录,确保网络安全与效率。通过C#代码示例展示了如何使用哈希表存储和管理用户的登录时间、访问IP及操作行为等信息,实现快速的查找、插入和删除操作。哈希表的应用显著提升了系统的实时性和准确性,尽管存在哈希冲突等问题,但通过合理设计哈希函数和冲突解决策略,可以确保系统稳定运行,为企业提供有力的安全保障。
|
1月前
|
存储 算法
深入解析PID控制算法:从理论到实践的完整指南
前言 大家好,今天我们介绍一下经典控制理论中的PID控制算法,并着重讲解该算法的编码实现,为实现后续的倒立摆样例内容做准备。 众所周知,掌握了 PID ,就相当于进入了控制工程的大门,也能为更高阶的控制理论学习打下基础。 在很多的自动化控制领域。都会遇到PID控制算法,这种算法具有很好的控制模式,可以让系统具有很好的鲁棒性。 基本介绍 PID 深入理解 (1)闭环控制系统:讲解 PID 之前,我们先解释什么是闭环控制系统。简单说就是一个有输入有输出的系统,输入能影响输出。一般情况下,人们也称输出为反馈,因此也叫闭环反馈控制系统。比如恒温水池,输入就是加热功率,输出就是水温度;比如冷库,
400 15
|
5天前
|
算法 数据安全/隐私保护 计算机视觉
基于Retinex算法的图像去雾matlab仿真
本项目展示了基于Retinex算法的图像去雾技术。完整程序运行效果无水印,使用Matlab2022a开发。核心代码包含详细中文注释和操作步骤视频。Retinex理论由Edwin Land提出,旨在分离图像的光照和反射分量,增强图像对比度、颜色和细节,尤其在雾天条件下表现优异,有效解决图像去雾问题。
|
5天前
|
算法 数据可视化 安全
基于DWA优化算法的机器人路径规划matlab仿真
本项目基于DWA优化算法实现机器人路径规划的MATLAB仿真,适用于动态环境下的自主导航。使用MATLAB2022A版本运行,展示路径规划和预测结果。核心代码通过散点图和轨迹图可视化路径点及预测路径。DWA算法通过定义速度空间、采样候选动作并评估其优劣(目标方向性、障碍物距离、速度一致性),实时调整机器人运动参数,确保安全避障并接近目标。
|
15天前
|
算法 数据安全/隐私保护
室内障碍物射线追踪算法matlab模拟仿真
### 简介 本项目展示了室内障碍物射线追踪算法在无线通信中的应用。通过Matlab 2022a实现,包含完整程序运行效果(无水印),支持增加发射点和室内墙壁设置。核心代码配有详细中文注释及操作视频。该算法基于几何光学原理,模拟信号在复杂室内环境中的传播路径与强度,涵盖场景建模、射线发射、传播及接收点场强计算等步骤,为无线网络规划提供重要依据。
|
16天前
|
机器学习/深度学习 数据采集 算法
基于GA遗传优化的CNN-GRU-SAM网络时间序列回归预测算法matlab仿真
本项目基于MATLAB2022a实现时间序列预测,采用CNN-GRU-SAM网络结构。卷积层提取局部特征,GRU层处理长期依赖,自注意力机制捕捉全局特征。完整代码含中文注释和操作视频,运行效果无水印展示。算法通过数据归一化、种群初始化、适应度计算、个体更新等步骤优化网络参数,最终输出预测结果。适用于金融市场、气象预报等领域。
基于GA遗传优化的CNN-GRU-SAM网络时间序列回归预测算法matlab仿真

热门文章

最新文章

推荐镜像

更多