新书推介:《语义网技术体系》
作者:瞿裕忠,胡伟,程龚
   XML论坛     W3CHINA.ORG讨论区     计算机科学论坛     SOAChina论坛     Blog     开放翻译计划     新浪微博  
 
  • 首页
  • 登录
  • 注册
  • 软件下载
  • 资料下载
  • 核心成员
  • 帮助
  •   Add to Google

    >> We choose to study algorithmic problems,  not because they are easy,  but because they are hard.
    [返回] 中文XML论坛 - 专业的XML技术讨论区计算机理论与工程『 算法理论与分析 』 → 文本匹配算法(又名最长子序列)LCS源码--java实现 查看新帖用户列表

      发表一个新主题  发表一个新投票  回复主题  (订阅本版) 您是本帖的第 12926 个阅读者浏览上一篇主题  刷新本主题   树形显示贴子 浏览下一篇主题
     * 贴子主题: 文本匹配算法(又名最长子序列)LCS源码--java实现 举报  打印  推荐  IE收藏夹 
       本主题类别:     
     lemonutzf 帅哥哟,离线,有人找我吗?
      
      
      等级:大一新生
      文章:1
      积分:57
      门派:GOOGLEBBS.NET
      注册:2006/8/4

    姓名:(无权查看)
    城市:(无权查看)
    院校:(无权查看)
    给lemonutzf发送一个短消息 把lemonutzf加入好友 查看lemonutzf的个人资料 搜索lemonutzf在『 算法理论与分析 』的所有贴子 引用回复这个贴子 回复这个贴子 查看lemonutzf的博客楼主
    发贴心情 文本匹配算法(又名最长子序列)LCS源码--java实现

    最近无聊写了个复读机程序,其中有一块要用到文本匹配, 于是有了以下代码. 在这里共享出来,也许有人用的到.  第一个实现(也就是代码中的 LCS 函数)是用的经典的矩阵实现(动态规划), 在一般的数据结构或算法课本上都可以找到对此算法的描述. 可是这个算法的时间和空间复杂度太高,都是N^2级别的. 在我的程序中有点不可忍受. 于是有了 LCS_DN 这个算法,它把时间和空间复杂度缩小到D*N(D是两个文本的差异), 所以当文本相差不大时,效率会挺高不少. LCS_DN_refined是改进版,空间复杂度虽然还是D*N,但减少了几个系数量.
          这些代码都可以直接拿过来用的, 界面也挺简单, 现对算法的使用略做解释:

    LCS是 Longest Common Subsequence 的缩写, 即最长公共子序列. 比如
    A = a1a2.........an
    B = b1b2.........bm
    如果有 ai1ai2...aik = bj1bj2....bjk 并且 i1<i2<...<ik AND j1<j2<..<jk
    那么 ai1ai2...aik  就是 A,B的一个公共子序列.  相应的最长公共子序列就是k最大的时候了.

    LCS可以用来计算两个文本的相似程度,以及生物学上的基因比较等实际应用。 在我的程序中它用来找出两个文本中不同的块. 比如, 我有一个英文听写, 然后还有一个原文. 我想知道我的听写都是哪些地方出错了, 就要同原文就行比较.  LCS就可做到这些.

    源码中的函数 LCS(String[] txt1, String[] txt2).  其中txt1,txt2分别是英文听写和原文, 用数组表示, 每个数组元素就是一个单词.  返回结果是一个MatchPair的数组, MatchPair用来存放匹配的位置. 比如最大公共子序列是 ai1ai2...aik = bj1bj2....bjk , 那么返回结果就是(i1,j1) (i2,j2) ...  (ik,jk).

    其他两个函数 LCS_ND LCS_ND_refined 实现了同样的功能. 并且效率也挺高不少. 如果程序需要,建议用LCS_ND_refined

    呵呵, 下面贴代码( 因为版式问题,代码很乱,可以考到eclipse或其他编辑器里再看)


    public final class TextUtil {

     public static class MatchPair {
      public int x;

      public int y;

      MatchPair(int x, int y) {
       this.x = x;
       this.y = y;
      }
     }

     public static MatchPair[] LCS(String[] txt1, String[] txt2) {

      int[][] matchCount = null;
      int[][] direction = null;

      matchCount = new int[txt1.length + 1][txt2.length + 1];
      direction = new int[txt1.length + 1][txt2.length + 1];

      for (int i = 0; i <= txt1.length; ++i) {
       matchCount[i][0] = 0;
       direction[i][0] = 0;
      }

      for (int i = 0; i <= txt2.length; ++i) {
       matchCount[0][i] = 0;
       direction[0][i] = 0;
      }

      for (int plus = 2; plus <= txt1.length + txt2.length; ++plus) {
       for (int x = Math.max(1, plus - txt2.length); x <= Math.min(
         plus - 1, txt1.length); ++x) {
        int y = plus - x;
        int count = 0;
        int dir = 0;
        if (txt1[x - 1].equals(txt2[y - 1])) {
         count = matchCount[x - 1][y - 1] + 1;
         dir = 2;
        } else if (matchCount[x - 1][y] > matchCount[x][y - 1]) {
         count = matchCount[x - 1][y];
         dir = 1;
        } else {
         count = matchCount[x][y - 1];
         dir = 3;
        }

        matchCount[x][y] = count;
        direction[x][y] = dir;
       }
      }

      int x = txt1.length, y = txt2.length;
      ArrayList ret = new ArrayList();
      int dir = direction[x][y];
      while (dir != 0) {

       if (dir == 2) {
        ret.add(new MatchPair(x - 1, y - 1));
        --x;
        --y;
       } else if (dir == 1) {
        --x;
       } else {
        --y;
       }

       dir = direction[x][y];
      }

      Collections.reverse(ret);
      return (MatchPair[]) ret.toArray(new MatchPair[0]);
     }

     public static MatchPair[] LCS_DN(String[] txt1, String[] txt2) {
      ArrayList list = new ArrayList();

      int M = txt1.length;
      int N = txt2.length;
      int MAX = M + N;

      int[] cur = null;
      int[] pre = new int[2 * MAX + 1];

      pre[1 + MAX] = 0;

      int x, y;
      for (int D = 0; D <= MAX; ++D) {
       list.add(pre);
       cur = new int[2 * MAX + 1];

       for (int k = -1 * D; k <= D; k += 2) {
        if (k == -1 * D || k != D
          && pre[k - 1 + MAX] < pre[k + 1 + MAX]) {
         x = pre[k + 1 + MAX];
        } else {
         x = pre[k - 1 + MAX] + 1;
        }
        y = x - k;
        while (x < txt1.length && y < txt2.length
          && txt1[x].equals(txt2[y])) {
         ++x;
         ++y;
        }
        cur[k + MAX] = x;

        if (x == txt1.length && y == txt2.length) {
         int x1, k1;
         ArrayList ret = new ArrayList();
         do {
          pre = (int[]) list.get(D);
          if (k == -1 * D || k != D
            && pre[k - 1 + MAX] < pre[k + 1 + MAX]) {
           x1 = pre[k + 1 + MAX];
           k1 = k + 1;
          } else {
           x1 = pre[k - 1 + MAX];
           k1 = k - 1;
          }

          while (x > x1 && (x - k) > (x1 - k1)) {
           --x;
           System.out.println("(" + x + "," + (x - k) + ")");
           ret.add(new MatchPair(x, x - k));
          }
          k = k1;
          x = x1;

         } while (--D >= 0);

         Collections.reverse(ret);
         return (MatchPair[]) ret.toArray(new MatchPair[0]);
        }
       }
       pre = cur;

      }
      return null;
     }

     public static MatchPair[] LCS_DN_refined(String[] txt1, String[] txt2) {
      ArrayList list = new ArrayList();

      int M = txt1.length;
      int N = txt2.length;
      int MAX = M + N;

      int[] cur = null;
      int[] pre = new int[1];

      pre[0] = 0;

      int x, y;
      for (int D = 0; D <= MAX; ++D) {
       list.add(pre);
       cur = new int[D + 1];

       for (int k = -1 * D; k <= D; k += 2) {
        if (k == -1 * D
          || k != D
          && pre[(k - 1 + D - 1) >> 1] < pre[(k + 1 + D - 1) >> 1]) {
         x = pre[(k + 1 + D - 1) >> 1];
        } else {
         x = pre[(k - 1 + D - 1) >> 1] + 1;
        }
        y = x - k;
        while (x < txt1.length && y < txt2.length
          && txt1[x].equals(txt2[y])) {
         ++x;
         ++y;
        }
        cur[(k + D) >> 1] = x;

        if (x == txt1.length && y == txt2.length) {
         int x1, k1;
         ArrayList ret = new ArrayList();

         do {
          pre = (int[]) list.get(D);
          if (k == -1 * D
            || k != D
            && pre[(k - 1 + D - 1) >> 1] < pre[(k + 1 + D - 1) >> 1]) {
           x1 = pre[(k + 1 + D - 1) >> 1];
           k1 = k + 1;
          } else {
           x1 = pre[(k - 1 + D - 1) >> 1];
           k1 = k - 1;
          }

          while (x > x1 && (x - k) > (x1 - k1)) {
           --x;
           System.out.println("(" + x + "," + (x - k) + ")");
           ret.add(new MatchPair(x, x - k));
          }
          k = k1;
          x = x1;

         } while (--D >= 0);

         Collections.reverse(ret);
         return (MatchPair[]) ret.toArray(new MatchPair[0]);
        }
       }

       pre = cur;

      }
      return null;
     }
    }


       收藏   分享  
    顶(0)
      




    点击查看用户来源及管理<br>发贴IP:*.*.*.* 2006/8/4 10:44:00
     
     limao1358 帅哥哟,离线,有人找我吗?
      
      
      等级:大一新生
      文章:0
      积分:56
      门派:XML.ORG.CN
      注册:2006/6/1

    姓名:(无权查看)
    城市:(无权查看)
    院校:(无权查看)
    给limao1358发送一个短消息 把limao1358加入好友 查看limao1358的个人资料 搜索limao1358在『 算法理论与分析 』的所有贴子 引用回复这个贴子 回复这个贴子 查看limao1358的博客2
    发贴心情 
    给个关于优化方法的说明啊!!!!!!!!!
    点击查看用户来源及管理<br>发贴IP:*.*.*.* 2007/7/26 18:34:00
     
     sunveronica 美女呀,离线,快来找我吧!
      
      
      等级:大一新生
      文章:1
      积分:55
      门派:XML.ORG.CN
      注册:2009/12/24

    姓名:(无权查看)
    城市:(无权查看)
    院校:(无权查看)
    给sunveronica发送一个短消息 把sunveronica加入好友 查看sunveronica的个人资料 搜索sunveronica在『 算法理论与分析 』的所有贴子 引用回复这个贴子 回复这个贴子 查看sunveronica的博客3
    发贴心情 
    是用的“基于位运算的最长公共子序列算法”优化的吗?
    点击查看用户来源及管理<br>发贴IP:*.*.*.* 2009/12/24 16:28:00
     
     GoogleAdSense
      
      
      等级:大一新生
      文章:1
      积分:50
      门派:无门无派
      院校:未填写
      注册:2007-01-01
    给Google AdSense发送一个短消息 把Google AdSense加入好友 查看Google AdSense的个人资料 搜索Google AdSense在『 算法理论与分析 』的所有贴子 访问Google AdSense的主页 引用回复这个贴子 回复这个贴子 查看Google AdSense的博客广告
    2025/1/21 18:00:10

    本主题贴数3,分页: [1]

    管理选项修改tag | 锁定 | 解锁 | 提升 | 删除 | 移动 | 固顶 | 总固顶 | 奖励 | 惩罚 | 发布公告
    W3C Contributing Supporter! W 3 C h i n a ( since 2003 ) 旗 下 站 点
    苏ICP备05006046号《全国人大常委会关于维护互联网安全的决定》《计算机信息网络国际联网安全保护管理办法》
    109.375ms