有这样几类问题:
建立数学模型,在有限时间内,用解析的方法求解
建立数学模型,但在有限时间内,用数学解析的方法求解困难,只好用搜索或模拟来求解,常见方法有:
穷举
深度优先搜索方法
广度优先搜索方法
启发式算法
穷举,适用于:
可预见确定解元素个数,且问题规模不是特别大
对于每个解变量a1,…,an的可能值为一个连续的值域
2)深度优先搜索,适用于:
求解初始结点到目标结点的所有方案
求解初始结点到目标结点的一种方案
3)广度优先搜索,适用于:
求解初始结点到目标结点的所有方案
求解初始结点所能达到的所有结点
求某一结点到某目标结点的最短路径
关于搜索算法
搜索是平常最常用的一种算法,同时应该也是最难掌握的一种算法。搜索类似于枚举、穷举。顾名思义就是将所有的情况都试一下(这个“试”是有逻辑的顺序的),当存在某种状态符合问题要求的时候,那个这个状态就是问题一个解。
搜索算法的效率是非常低的,时间复杂度一般都是级数增长的。所以在搜索算法的同时我们通常要加上优化,例如:剪枝,记忆化搜索,分枝定界,以降低时间复杂度。
纯随机搜索(Random Walk)
广度优先搜索(BFS)
深度优先搜索(DFS)
启发式搜索:启发式搜索就是:在状态空间中,对每一个搜索的位置进行评估,得到最好的位置,再从这个位置进行搜索直到目标(即:通过启发式函数,选择代价最少的结点作为下一步搜索结点而跳转其上),使搜索过程沿着被认为最有希望的前沿区段发展。
回溯法:
有许多问题,当需要找出它的解集或者要求回答什么解是满足某些约束条件的最佳解时,往往要使用回溯法。
回溯法的基本做法是搜索,是一种组织得井井有条的,能避免不必要搜索的穷举式搜索法。这种方法适用于解一些组合数相当大的问题。
回溯法在问题的解空间树中,按深度优先策略,从根结点出发搜索解空间树。算法搜索至解空间树的任意一点时,先判断该结点是否包含问题的解。如果肯定不包含,则跳过对该结点为根的子树的搜索,逐层向其祖先结点回溯;否则,进入该子树,继续按深度优先策略搜索。
这种以“深度优先”的方式系统地搜索问题地解地算法称为回溯法。
采用回溯法,必须要要弄清如下问题:
1)问题的解空间树或解空间图是什么?是子集树、或排列树,还是m叉树,或更复杂的树或图?
2)搜索的方向是:深度优先
3)用剪枝函数来避免不可能产生最优解的分支的搜索,越好的剪枝和限界函数可以砍去更多的无用分支搜索。
问题的解空间:
问题的解向量:回溯法希望一个问题的解能够表示成一个n元式(x1,x2,…,xn)的形式。
显约束:对分量xi的取值限定。
隐约束:为满足问题的解而对不同分量之间施加的约束。
解空间:对于问题的一个实例,解向量满足显式约束条件的所有多元组,构成了该实例的一个解空间。
注意:有时同一个问题可以有多种解空间表示,有些表示更简单,所需表示的状态空间也更小。(存储量少,搜索方法简单)
生成问题状态的基本方法:
扩展结点:一个正在产生儿子的结点称为扩展结点
活结点:一个自身已生成但其儿子还没有全部生成的节点称做活结点
死结点:一个所有儿子已经产生的结点称做死结点
深度优先的问题状态生成法:如果对一个扩展结点R,一旦产生了它的一个儿子C,就把C当做新的扩展结点。在完成对子树C(以C为根的子树)的穷尽搜索之后,将R重新变成扩展结点,继续生成R的下一个儿子(如果存在)
广度优先的问题状态生成法:在一个扩展结点变成死结点之前,它一直是扩展结点
回溯法:为了避免生成那些不可能产生最佳解的问题状态,要不断地利用剪枝函数来处死那些实际上不可能产生所需最优解的活结点,以减少问题的计算量。具有剪枝函数的深度优先生成法称为回溯法
基本思想:
回溯算法框架=
问题的解空间+深度优先遍历+判断结果的函数+剪枝函数
这个剪枝函数:
用约束函数在扩展结点处剪去不满足约束的子树;
用限界函数剪去得不到最优解的子树。
回溯法对解空间作深度优先搜索,因此,在一般情况下用递归方法实现回溯法。
void backtrack (int t)
{
if (t>n) output(x);
else
for (int i=f(n,t);i<=g(n,t);i++) {
x[t]=h(i);
if (constraint(t)&&bound(t)) backtrack(t+1);
}
}
采用树的非递归深度优先遍历算法,可将回溯法表示为一个非递归迭代过程。
void iterativeBacktrack ()
{
int t=1;
while (t>0) {
if (f(n,t)<=g(n,t))
for (int i=f(n,t);i<=g(n,t);i++) {
x[t]=h(i);
if (constraint(t)&&bound(t)) {
if (solution(t)) output(x);
else t++;
} //end if
} //end for
else t–;
} //end while
}