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. NSArray block用法

    28.使用block 块遍历整个数组.这个block 需要三个参数,id obj 表示数组中的元素. NSUInteger idx 标示元素的下标, bool *stop 是一个bool类型的参数. ...

  2. ASP 中调用函数关于Call使用注意的问题

    Function TestFun(Tstr) TStr = "Fun2" End Function Sub TestSub(TStr) Tstr = "Sub2" ...

  3. hdu 2999 Stone Game, Why are you always there? 博弈论

    SG函数应用!! 代码如下: #include<cstdio> #include<cstring> #include<iostream> #include<c ...

  4. spring PropertyPlaceholderConfigurer 找不到配置文件原因

    1:  spring 版本问题 参见: http://www.cnblogs.com/alex-blog/archive/2012/12/25/2832357.html 2: bean id 同名   ...

  5. JSTL Tag学习笔记(二)之<fmt: />

    JSTL的formatting tags可以用来格式化和显示文本.日期.时间.数字.如果在JSP页面中要用到该库提供的tag的话,需要引入如下taglib: <%@ taglib prefix= ...

  6. [iOS]集成环信SDK然后运行时候crash了-[NSBundle initWithURL:]: nil URL argument'

    Crash的reason是-[NSBundle initWithURL:]: nil URL argument' 1.首先我是用cocoapods导入的环信的SDK.然后怎么运行怎么crash. 2. ...

  7. HttpClient使用详解(转)

     HttpClient使用详解 Http协议的重要性相信不用我多说了,HttpClient相比传统JDK自带的URLConnection,增加了易用性和灵活性(具体区别,日后我们再讨论),它不仅是客户 ...

  8. JavaScript获取DOM元素位置和尺寸大小

      在一些复杂的页面中经常会用JavaScript处理一些DOM元素的动态效果,这种时候我们经常会用到一些元素位置和尺寸的计算,浏览器兼容性问题也是不可忽略的一部分,要想写出预想效果的JavaScri ...

  9. Oracle ->> Oracle下查看实际执行计划的方法

    也许有很多种方法,这里只是书上学到的一种方法 with a as ( order by grp_factor) t ) select b.id, a.grp_factor )b --use v$sql ...

  10. Git设置及GitHub的使用

    把github上的help略略翻译一遍.备忘. First : 安装:ubuntu 下,终端输入命令: sudo apt-get install git-core git-gui git-doc Ne ...