有趣的算法问题之巧思妙想

有趣的算法

算法之所以有趣,在于他能够化繁为简,他能概括统御世间万物,将一个复杂的问题归结为一个非常简单的问题。其实所有高阶的算法,都可以用两个大的方法去解决,而且屡试不爽。分别是动态规划贪心算法.
我们先从一道动态规划的问题先说起吧。

动态规划问题

最长公共子序列问题(LCS),给定两个序列X和Y,求他们之间最长的公共子序列。

哇,这题目感觉有点难啊。最长公共子序列问题其实可以归结为最最最最最简单的一个递推表达式,首先我们假设序列X元素个数为i,Y为j,C[i, j]表示最长公共子序列的长度。好那么问题就变成了,如何求取这个最长公共子序列的长度问题。公式也是非常简单 ==easy== :smile:

$$C[i, j] = \left{ \begin{array}{lr} 0,\quad if \ x=0 , or \ y=0 \ C[i-1, j-1] + 1, \quad if ; i,j>0 ; , x_i=y_i \ max{C[i, j-1], C[i-1, j]} , \quad if \ i,j>0, \ x_i\neq x_j \end{array} \right. $$

这个公式还是非常难打啊,大概就是这么一个意思吧。咋一看这个表达式很牛逼啊,你随便给我两个序列,我通过他就可以求出最长子序列的长度?闲话不多说,让我们直接上代码看一下到底有没有这么牛逼。

假如我们有两个序列,我们用字符串来表示吧 X=cnblogs, Y=belong, 肉眼可以看到最长子序列为blog。那么长度就是4,让我们看看到底有没有这么牛逼。

在这之前,我得理清一下思路,首先是这样的,我们求取的这个C[i, j],实际上就是一个二维的矩阵,你想啊i是从0变到i,j是从0变到j,那一路求过来,不就是一个i行j列的矩阵吗?目标值就是最右下角的这个元素。既然如此那么变成就好办了。

// this code calculate the max length of common sub-sequence of 2 strings
int lcs_length(string a, string b) {
    // given 2 string, return the LCS length
    // define a 2 dim array
    int matrix[a.size()+1][b.size()+1];
    for (int i = 0; i <= a.size(); ++i) {
        matrix[i][0] = 0;
    }
    for (int j = 0; j <= b.size(); ++j) {
        matrix[0][j] = 0;
    }

    for(int i = 1; i <= a.size(); i++) {
        for(int j = 1; j <= b.size(); j++) {
            if (a[i-1] == b[j-1]) {
                matrix[i][j] = matrix[i-1][j-1] + 1;
            } else {
                matrix[i][j] = matrix[i][j-1] > matrix[i-1][j] ? matrix[i][j-1] : matrix[i-1][j];
            }
        }
    }
    return matrix[a.size()][b.size()];
}

很显然,这个函数可以正确的得到最长公共子串的长度。看上去还是很牛逼,但是其实道理也非常简单,无外乎就是上面的三个公式。那么你可能回问了,上面三个公式是怎么来的呢?其实就是一个非常简单的递推,假如说公共子串Z的最后一个元素是X的最后一个元素,那么肯定也是Y的最后元素,那如果将X去掉最后元素,Y去掉最后一个元素,最长公共子串就是去掉之后的+1,就是加去掉的这个嘛。那如果说最后一个元素都不是X, Y的最后元素,那更好办了,这个时候公共子串就是X和Y的中间某一个子串嘛,这个时候X去掉最后一个,再来求公共子串,还是一样啊,或者Y去掉一个,也是一样啊,就直接就等于X或者Y去掉一个的共同子串的最大值了。(有人会问,为什么不等于X,Y都去掉一个的最大值呢?也就是 $$ max{C[i-1, j-1], C[i-1, j-1]}$$, 这是不行的,原因很简单,你X去掉一个之后,最长子串就有可能包含Y的最后一个值了,你都去掉会减少很多种情况,不可取 )。

这个问题我们已经完成了历史性的一步: 可以求取两个序列的最长子序列的长度。, 那么下一步就是,怎么找到这个最长子序列。这一步思路是这样的:

(你可能无法想象,我完成求最大公共子序列上调试C++代码踩了一下午坑,我曹,真的是天坑)。先说一下思路吧,非常复杂,首先在上面的函数里面我们给他传入一个pFlag的二维数组,注意这是一个指针,因为后面需要递归遍历他。在这个二维数组里面存储的大小和matrix是一样,只不过这里面存储的都是字符串,为了便于理解我存储为 : “left", "up", "left_up",三种字符串,其实你如果自己画了matrix这个表,就会发现,其实可以通过这样的箭头去回溯这个最长子序列是什么,你会发现恰恰是箭头所指向的路径。然后我们用一个函数递归,根据箭头来找到对应的子序列。所有代码如下:

#include <iostream>
#include <vector>
using namespace std;
void sub_sequence(int i, int j, string **pFlag, string a) {
    if (i == 0 || j == 0) {
        return;
    }
    if (pFlag[i][j] == "left_up") {
        sub_sequence(i - 1, j - 1, pFlag, a);
        cout << a[i-1] << " ";
    } else {
        if (pFlag[i][j] == "left") {
            sub_sequence(i, j-1, pFlag, a);
        } else {
            sub_sequence(i-1, j, pFlag, a);
        }
    }
}

int lcs_length(string a, string b, string **pFlag) {
    // given 2 string, return the LCS length
    // define a 2 dim array
    int matrix[a.size()+1][b.size()+1];
    for (int i = 0; i <= a.size(); ++i) {
        matrix[i][0] = 0;
    }
    for (int j = 0; j <= b.size(); ++j) {
        matrix[0][j] = 0;
    }
    for(int i = 1; i <= a.size(); i++) {
        for(int j = 1; j <= b.size(); j++) {
            if (a[i-1] == b[j-1]) {
                matrix[i][j] = matrix[i-1][j-1] + 1;
                // using string to indicate location
                pFlag[i][j] = "left_up";
            } else {
                if (matrix[i][j-1] > matrix[i-1][j]) {
                    matrix[i][j] = matrix[i][j-1];
                    pFlag[i][j] = "left";
                } else {
                    matrix[i][j] = matrix[i-1][j];
                    pFlag[i][j] = "up";
                }
            }
        }
    }
    return matrix[a.size()][b.size()];
}

int main()
{
    string b = "gheteuponthiop";
    string a = "giothuphyo";
    // 这里应该是**pFlag, markdown渲染有问题,二维指针
    auto ** pFlag = new string* [a.size() + 1];
    for (int k = 0; k <= a.size(); ++k) {
        pFlag[k] = new string[b.size() + 1];
    }
    int l = lcs_length(a, b, pFlag);
    sub_sequence((int) a.size(), (int) b.size(), pFlag, a);
    cout << endl;
    cout << l << endl;
    return 0;
}

收工!以后遇到求最大公共子序列问题就来我的博客!!!!

写到这里发现并不是那么有趣了。很复杂啊。。。不过坚信那句,万变不离其宗。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,569评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,499评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,271评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,087评论 0 209
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,474评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,670评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,911评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,636评论 0 202
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,397评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,607评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,093评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,418评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,074评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,092评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,865评论 0 196
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,726评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,627评论 2 270

推荐阅读更多精彩内容

  • 回溯算法 回溯法:也称为试探法,它并不考虑问题规模的大小,而是从问题的最明显的最小规模开始逐步求解出可能的答案,并...
    fredal阅读 13,509评论 0 89
  • 背景 一年多以前我在知乎上答了有关LeetCode的问题, 分享了一些自己做题目的经验。 张土汪:刷leetcod...
    土汪阅读 12,662评论 0 33
  • 动态规划(Dynamic Programming) 本文包括: 动态规划定义 状态转移方程 动态规划算法步骤 最长...
    廖少少阅读 3,187评论 0 18
  • 定义一组子问题,按照由小到大,以小问题的解答支持大问题求解的模式,依次解决所有的子问题,并最终得到原问题的解答。 ...
    芥丶未央阅读 811评论 0 2
  • 感谢前面自己的不温不火,才可为现在的自己添一把干柴烈火。 读书本身就是一个顿悟的过程,好多事情慢慢想就会通透很多。...
    一只特例独行的鈺猪阅读 117评论 0 0