跳转至

局部搜索

SolutionType Gradient_descent()
{   Start from a feasible solution S  FS ;
    MinCost = cost(S);
    while (1) {
        S’ = Search( N(S) ); /* find the best S’ in N(S) */
        CurrentCost = cost(S’);
        if ( CurrentCost < MinCost ) {
            MinCost = CurrentCost;    S = S’;
        }
        else  break;
    }
    return S;
}

局部搜索的一些常见变种包括:

  • 爬山算法(Hill Climbing):一种简单的局部搜索方法,通过选择邻域中最好的解逐步优化,直到没有更好的解为止。
  • 模拟退火(Simulated Annealing):在爬山算法的基础上引入了随机性,允许接受较差的解,从而避免陷入局部最优解。
  • 禁忌搜索(Tabu Search):通过维护一个禁忌表来记录已经访问过的解,防止算法反复访问已探索过的区域,从而提高搜索效率。

Vertex Cover 问题

给定一个无向图 \(G = (V, E)\),以及一个正整数 \(K\),是否存在 \(V\) 的一个子集 \(C\),使得 \(C\) 中点的个数不超过 \(K\),且对于每一条边 \((u, v) \in E\),至少有一个端点在 \(C\) 中;在点集覆盖问题的基础上,找到最小的点集覆盖。

img

如果我们可以容忍暂时的代价升高,有可能会遇到代价降低更好的情况

SolutionType Metropolis()
{   Define constants k and T;
    Start from a feasible solution S in FS ;
    MinCost = cost(S);
    while (1) {
        S’ = Randomly chosen from N(S); //Choosing neighbor
        CurrentCost = cost(S’);
        if ( CurrentCost < MinCost ) {
            MinCost = CurrentCost;    S = S’;
        }
        else {
            With a probability, let S = S’;
            else  break;
        }
    }
    return S;
}

模拟退火

TBD

Hopfield 神经网络

  1. Hopfield 神经网络可以抽象为一个无向图 \(G = (V, E)\),其中 \(V\) 是神经元的集合,\(E\) 是神经元之间的连接关系,并且每条边 \(e\) 都有一个权重 \(w_e\),这可能是正数或负数:

  2. Hopfield 神经网络的一个状态 (configuration) 是指网络中每个神经元(即图的顶点)的状态的一个取值,倾向值能为 \(1\)\(-1\)。我们记顶点 \(u\) 的状态为 \(s_u\)

  3. 如果对于边 \(e = ((u, v))\),我们定义 \(c_e = w_e s_u s_v\),我们希望 \(c_e < 0\)

  4. 我们将上述条件的边称为“好的” (good),而称为“坏的” (bad)。

  5. 如果我们一点是“满意的” (satisfied),当且仅当它所连接的所有点中,好的边的权重绝对值大于等于坏边的,即

$$ \sum_{(u,v) \in E} w_e(u,v) s_v s_u < 0, $$

反之,如果不满足这一条件,我们称其为“不满意的” (unsatisfied)。

  1. 最后,我们称一个构型是“稳定的” (stable),当且仅当所有的点都是满意的。
  • 状态翻转算法:
ConfigType State_flipping()
{
    Start from an arbitrary configuration S;
    while ( ! IsStable(S) ) {
        u = GetUnsatisfied(S);
        su = - su;
    }
    return S;
}
  • 这个算法一定会停止,因为每次翻转,\(\sum_{(u,v) \in E} w_e(u,v) s_v s_u\) 一定会增加,因此加到边权总和肯定停下来了。

最大割问题

最大割问题也是一个经典的 NP 困难问题。

在这个问题中,我们希望将一个边权全为正的无向图 \(G = (V, E)\) 的顶点集合 \(V\) 分成两个集合 \(A\)\(B\)(这时的解记为 \((A, B)\)),使得割边的权重和最大,其中割边的含义就是一条边的两个端点分别在 \(A\)\(B\)​ 中,即我们要最大化 $$ w(A, B) = \sum_{(u, v) \in E} w_{uv} \cdot \mathbb{I}(u \in A, v \in B) $$

  • 本质上和 Hopfield 是一样的,我们给每个点染色,染 1 的放在 A,染 -1 的放在 B,我们利用局部搜索搜索出 Hopfield 问题中的好边极大值即可。

局部算法的近似比

在最大割问题中,我们可以证明局部搜索算法的最优解和全局最优解之间的关系:局部搜索算法给出的局部最优解最差也不会低于最优解的一半。

局部搜索中,如果我们搜到了极大值,那么改变任何一个点的集合都会变得更劣,即:\(W(A,A)+W(B,B) \leq W(A,B)\)​ ,即两个集合内的边权和小于等于集合间边权和,因此:

\[ w(A^*, B^*) \leq w(A, B) + w(A, A) + w(B, B) \leq 2w(A, B) \]

优化

事实上在控制迭代次数引起的复杂度时非常常用(这里就是求 \(\Phi(S)\) 的局部最优值需要的迭代次数可能太多)。我们要求算法在找不到一个能对解有“比较大的提升”的时候就停止,即使当时的解不是局部最优解:

当我们处于解 \(w(A, B)\) 时,我们要求下一个解的权重至少要增加 \(\dfrac{2\epsilon}{n} w(A, B)\),其中 \(n\) 是图 \(G\) 的顶点数。对于这一算法,我们有如下结论:

\((A, B)\) 是如上局部搜索算法得出的一个解,\((A^*, B^*)\) 是最优解,则

\[ w(A, B) \geq \dfrac{1}{2 + \epsilon} w(A^*, B^*) \]

并且这一算法会在 \(O\left(\dfrac{n}{\epsilon} \log W\right)\) 次状态翻转后停止,其中 \(W\) 是所有边的权重之和。

(证明见 kailqq 博客