package DataStructure;

import java.util.ArrayList;
import java.util.List; //KMP算法的实现
//以下代码由freedom结合资料理解写出
public class DMPtest1 {
private int next[] ;
private String target; //主串
private String pattern;//子串
char[] t; //主串字符
char[] p; //子串字符
private static List<Integer> list ; public DMPtest1() {}
public DMPtest1(String pattern,String target) {
this.pattern = pattern;
this.target = target;
p = this.pattern.toCharArray();
t = this.target.toCharArray();
//list = new ArrayList<Integer>(); //初始化一个集合用于保存匹配的子串在主串中的索引
next = this.getNextArray(this.pattern); //接受返回的已经算好的数组 } public void displayNext() {
//打印next数组
for(int i=0;i<next.length; i++) {
System.out.print(next[i]);
} } public List<Integer> getTargetIndex() {
List<Integer> list = new ArrayList<Integer>();
int num=0;
int i = 0;//主串target下标
int j = 0;//子串pattern下标
if(t.length>=p.length) { while(i<t.length) { //因为主串的下标是不变的,只有匹配超过他自己长度就会跳出循环 if(p[j]==t[i]) {
num++; //第一次相等的话那么num清零
j++;
i++; //System.out.println("已经找到一个目标"+j+p.length);
if(j>p.length-1) {//若已经是最后一个数了 list.add(i-j); //把匹配的字符串在主串中位置(第一个字符的索引)添加到list
j=0; //说明一个子串已经匹配完毕
//j已经等于p数组的最后一个下标的下一个下标了
}
}else {
//两字符不匹配的话
//j的新索引直接等于next[j]结论在纸上写着 if(num==0) {
j=0;
//第一次不相等,那么 i 需要自加
i++;
//这种结构要小心,一定要写在下一个if的前面!
} if(num>0) { //说明不是第一次不相等,那么主串下标不用自加
j=next[j];
num=0; } //System.out.println(j);
}
} }else {
System.out.println("模式串必须 大于或等于 主串 !");
}
return list;
} /**
* 1 、该方法 用于返回一个next[]数组,保存的是模式串的相应T{0~j-1}字串的(最长前缀和后缀相同匹配字符的数量)j是模式串的下标
*
* 2 、 pattern是模式字符串,要转化成char[],如你要搜索freedom,就要把freedom,转成f,r,e,e,d,o,m的char类型数组
*
* 3、此方法是用递归的思想实现,可以一眼写出next数组。在后面会给解释!
*
*/
public int[] getNextArray(String pattern) { int next[] = new int[p.length];
next[0] = -1;
int j = 0; //next数组下标
int k = -1; // 用于临时保存next数组的值 //因为next数组求出来后目的是为了求DMP,所以把整个next数组向右平移1,所以第一位普遍是-1,代表没有该字串
//而next[1]=0,因为第二个位置的字串求得是T{0~j-1}的前缀和后缀最长相同值的长度,所以1个字符是空集,这个会在方法体里面实现 //当j给最后一个字符赋完值,就要跳出循环,如果该字符串有8个字符,那么j必须小于8-1
while(j<p.length-1) {
if(k==-1||p[k]==p[j]) {
//如果匹配成功,j,k两下标都要自加,以比较下一个组合是否相等
j++;
k++;
next[j] = k;
} else {
//如果不匹配,那么j依然不动,k取上一个k的值
k = next[k]; //知道k为-1没有匹配值返回0
}
}
return next;
}
public static void main(String args[]) {
DMPtest1 dmp = new DMPtest1("abc","afwefwaefaaaaabcawiefjawoijfeioawjofabc"); //为next数组初始化了
//dmp.displayNext();
list = dmp.getTargetIndex(); //遍历匹配主串,并将索引返回给list
for(int i=0;i<list.size();i++) {
System.out.println("字符串在主串中的位置是"+list.get(i));
} System.out.println("共有 {"+list.size()+"} 个匹配结果"); }
}

java实现的kmp算法的更多相关文章

  1. 算法(Java实现)—— KMP算法

    KMP算法 应用场景 字符串匹配问题 有一个字符串str1 = " hello hello llo hhello lloh helo" 一个子串str2 = "hello ...

  2. KMP算法-Java实现

    目的: 为了解决字符串模式匹配 历程: 朴素模式匹配:逐次进行比较 KMP算法:利用匹配失败得到的信息,来最大限度的移动模式串,以此来减少比较次数提高性能 概念: m:是目标串长度 n:是模式串长度 ...

  3. 经典KMP算法C++与Java实现代码

    前言: KMP算法是一种字符串匹配算法,由Knuth,Morris和Pratt同时发现(简称KMP算法).KMP算法的关键是利用匹配失败后的信息,尽量减少模式串与主串的匹配次数以达到快速匹配的目的.比 ...

  4. 大话数据结构(十二)java程序——KMP算法及改进的KMP算法实现

    1.朴素的模式匹配算法 朴素的模式匹配算法:就是对主串的每个字符作为子串开头,与要连接的字符串进行匹配.对主串做大循环,每个字符开头做T的长度的小循环,直到成功匹配或全部遍历完成为止. 又称BF算法 ...

  5. Java实现KMP算法

    /**  * Java实现KMP算法  *   * 思想:每当一趟匹配过程中出现字符比较不等,不需要回溯i指针,   * 而是利用已经得到的“部分匹配”的结果将模式向右“滑动”尽可能远   * 的一段 ...

  6. [转]KMP算法理解及java实现

    这大概是我看的最好懂的KMP算法讲解了,不过我还只弄懂了大概思想,算法实现我到时候用java实现一遍 出处:知乎 https://www.zhihu.com/question/21923021/ans ...

  7. KMP算法中next数组的理解与算法的实现(java语言)

    KMP 算法我们有写好的函数帮我们计算 Next 数组的值和 Nextval 数组的值,但是如果是考试,那就只能自己来手算这两个数组了,这里分享一下我的计算方法吧. 计算前缀 Next[i] 的值: ...

  8. Java数据结构之字符串模式匹配算法---KMP算法2

    直接接上篇上代码: //KMP算法 public class KMP { // 获取next数组的方法,根据给定的字符串求 public static int[] getNext(String sub ...

  9. Java数据结构之字符串模式匹配算法---KMP算法

    本文主要的思路都是参考http://kb.cnblogs.com/page/176818/ 如有冒犯请告知,多谢. 一.KMP算法 KMP算法可以在O(n+m)的时间数量级上完成串的模式匹配操作,其基 ...

随机推荐

  1. Android loader 详解

    装载器从android3.0开始引进.它使得在activity或fragment中异步加载数据变得简单.装载器具有如下特性: 它们对每个Activity和Fragment都有效. 他们提供了异步加载数 ...

  2. POJ 3468 A Simple Problem with Integers(线段树)

    题目链接 题意 : 给你n个数,进行两种操作,第一种是将a到b上所有元素都加上c,第二种是查询a到b上所有元素之和输出. 思路 : 线段树,以前写过博客,但是现在在重刷,风格改变,,所以重新写一篇.. ...

  3. POJ 1027 The Same Game(模拟)

    题目链接 题意 : 一个10×15的格子,有三种颜色的球,颜色相同且在同一片内的球叫做cluster(具体解释就是,两个球颜色相同且一个球可以通过上下左右到达另一个球,则这两个球属于同一个cluste ...

  4. ListView中EditText的数据加载错乱的问题

    我在ListView中用BaseAdapter的getView()方法加载适配器,每个Item里有一个TextView和一个EditText,当我在第一个EditText里面输入数据,比如1234时, ...

  5. JS初学的一些易错的知识点

    1.  false ,0 , "" ,undefined , null  在Boolean 环境下当成 false: null  在数值环境下当成 0: undefined 在数值 ...

  6. linux read命令详解

    read命令从键盘读取变量的值,通常用在shell脚本中与用户进行交互的场合.该命令可以一次读取多个变量的值,变量和输入的值都需要使用空格隔开. 语法 read(选项)(参数) 选项 -p:指定读取值 ...

  7. TCP非阻塞accept和非阻塞connect

    http://blog.chinaunix.net/uid-20751538-id-238260.html 非阻塞accept     当一个已完成的连接准备好被accept的时候,select会把监 ...

  8. CSS布局:让页底内容永远固定在底部

    我们在设计一些页面内容甚少的网页时(典型应用就是登陆页面),由于显示器的分辨率大,在正常情况下,假如页面内容高度小于浏览器高度时,页面底部以下会留下很大的空间... 先看示例:http://www.h ...

  9. 世界上还有一个东西叫Virtual Pascal

    官网是:http://web.archive.org/web/20060312064321/http://www.vpascal.com/news.php?item.16 不过2005年就不再维护了. ...

  10. Android 常用时间格式转换代码

    /** * 获取现在时间 * * @return 返回时间类型 yyyy-MM-dd HH:mm:ss */ public static Date getNowDate() { Date curren ...