一个简易的kmp教学并给出java实现
简单介绍一下问题
给定source字符串,找出target字符串出现的首位
例如
source 为“abddabddabc”
target 为 “abddabc”
从第一位开始比较 |a b d d a b|d d a b c
|a b d d a b|c 不匹配
从第二位继续比较 a|b d d a b d d a b c
|a 不匹配
。。。
。。。
从第五位继续比较 a b d d|a b d d a b c
a b d d a b c 匹配成功
再给出kmp算法
从第一位开始比较 a b d d a b|d d a b c
a b d d a b|c 不匹配
————————————————————————————————
观察失配处前两位ab与target的前两位ab一致,
而整个target不再出现ab,所以即使出现匹配,也只能出现在最后两位
也就是ab之后,所以我们可以直接跳过中间所有位
直接把target的前两位与source当前ab对齐,继续匹配
————————————————————————————————
第二次匹配 a b d d| a b|| d d a b c
| a b||d d a b c
其中单斜杠处即为所求 双斜杠处为第二次匹配起始位
在上个例子 两条长横线中的内容就是kmp的核心思路:
不需要每一次失配都只从下一位继续,
只需要对比首位的重合部分,移动target,使其首尾重合部分对其,即可继续对比
再举一个极端的例子,便于理解
target 为 abcdefgh
如果source存在匹配部分,那这部分必由a起始
所以在匹配的过程中,发生了失配,即可全部跳过,对比下一位是否为a
而对于首尾有重合部分,则需要从重合部分下一位开始,而重合部分的起始位即为所求
重新用自然语言描述一下我们要做的事:
1 开始匹配 如果完成匹配 返回重合的起始位
2 发生失配 从当前失配位 观察target的首尾重合部分
3 调整target到重合部分首位 从失配位继续匹配
而失配位前target的首尾重合部分的长度由target自身决定,举例说明
a b d d a b d d a b c
0 1 2 3 4 5 6 7 8 9 10
0 0 0 0 1 2 3 4 1 2 0
比如当source为
a b d d a b d d c c c c c c c c c
0 1 2 3 4 5 6 7 8 9 10
0 0 0 0 1 2 3 4 1 2 0
a b d d a b d d a b c
我们在第八位发生失配,那么在0+8-4位为起始,从target的4 位继续匹配即可
下面给出java代码实现:
public static void main(String[] args) {
String source="abddabkdabc";
String target="abddabc";
int a=Solution.strStr(source, target);
System.out.println(a);
}
public static int strStr(String source, String target) {
if(target=="") {
return 0;
}
if(source.length()<target.length()) {
return -1;
}
int[] next=nextKmp(target);
int j=0;
int i=0;
while(true){
while(j<target.length()&&(i+j)<source.length()&&source.charAt(i+j)==target.charAt(0+j)) {
//System.out.println(source.charAt(i+j));
j++;
}
if(j==target.length()) {
System.out.println("!");
return i;
}
if((i+j)==source.length()) {
return -1;
}
if(j==0) {
i++;
continue;
}
i=i+j-next[j-1];
j=next[j-1];
}
}
if(str.equals("")) {
return null;
}
int[] next=new int[str.length()];
for (int i = 0; i < next.length; i++) {
next[i]=0;
}
for (int i = 1; i < str.length(); ) {
int j=0;
while((i+j)<str.length()&&str.charAt(i+j)==str.charAt(0+j)) {
next[i+j]=j+1;
j++;
}
if(j==0) {
i++;
}else {
i+=j;
}
}
return next;
}
}
一个简易的kmp教学并给出java实现的更多相关文章
- Java实践:一个简易的http server和client的java源码学习和总结。
一.基本思路: 1.服务器端通过socket(), 监听在TCP 8080端口,等待客户端来连接. 2.服务器端解析客户端的HTTP请求中的URI值,把本地的目录下指定文件通过java的读取文件的方式 ...
- .NET Core的文件系统[5]:扩展文件系统构建一个简易版“云盘”
FileProvider构建了一个抽象文件系统,作为它的两个具体实现,PhysicalFileProvider和EmbeddedFileProvider则分别为我们构建了一个物理文件系统和程序集内嵌文 ...
- 自己来实现一个简易的OCR
来做个简易的字符识别 ,既然是简易的 那么我们就不能用任何的第三方库 .啥谷歌的 tesseract-ocr, opencv 之类的 那些玩意是叼 至少图像处理 机器视觉这类课题对我这种高中没毕业的人 ...
- 探秘Tomcat——一个简易的Servlet容器
即便再简陋的服务器也是服务器,今天就来循着书本的第二章来看看如何实现一个servlet容器. 背景知识 既然说到servlet容器这个名词,我们首先要了解它到底是什么. servlet 相比你或多或少 ...
- ENode 2.0 - 第一个真实案例剖析-一个简易论坛(Forum)
前言 经过不断的坚持和努力,ENode 2.0的第一个真实案例终于出来了.这个案例是一个简易的论坛,开发这个论坛的初衷是为了验证用ENode框架来开发一个真实项目的可行性.目前这个论坛在UI上是使用了 ...
- 使用MVVM框架avalon.js实现一个简易日历
最近在做公司内部的运营管理系统,因为与日历密切相关,同时无需触发条件直接显示在页面上,所以针对这样的功能场景,我就用avalon快速实现了一个简易日历,毕竟也是第一次造日历这种轮子,所以这里记录下我当 ...
- C 实现一个简易的Http服务器
引言 做一个老实人挺好的,至少还觉得自己挺老实的. 再分享一首 自己喜欢的诗人的一首 情景诗. 每个人总会有问题,至少喜欢就好, 本文 参照 http 协议 http://www.cnblogs. ...
- avalon.js实现一个简易日历
使用MVVM框架avalon.js实现一个简易日历 最近在做公司内部的运营管理系统,因为与日历密切相关,同时无需触发条件直接显示在页面上,所以针对这样的功能场景,我就用avalon快速实现了一个简 ...
- Tinychatserver: 一个简易的命令行群聊程序
这是学习网络编程后写的一个练手的小程序,可以帮助复习socket,I/O复用,非阻塞I/O等知识点. 通过回顾写的过程中遇到的问题的形式记录程序的关键点,最后给出完整程序代码. 0. 功能 编写一个简 ...
随机推荐
- 如何快速掌握DDT数据驱动测试?
1.前言 (网盗概念^-^)相同的测试脚本使用不同的测试数据来执行,测试数据和测试行为完全分离, 这样的测试脚本设计模式称为数据驱动.(网盗结束)当我们测试某个网站的登录功能时,我们往往会使用不同的用 ...
- elasticsearch常用命令备注
1.检查集群健康状态 curl 'localhost:9200/_cat/health?v' 2.检查节点健康状态 curl 'localhost:9200/_cat/nodes?v' 3.新增一条索 ...
- iOS开发之虾米音乐频道选择切换效果分析与实现
今天博客的内容比较简单,就是看一下虾米音乐首页中频道选择的一个动画效果的实现.之前用mask写过另外一种Tab切换的一种效果,网易云音乐里边的一种Tab切换效果,详情请移步于"视错觉:从一个 ...
- FFmpeg命令行工具学习(五):FFmpeg 调整音视频播放速度
FFmpeg对音频.视频播放速度的调整的原理不一样.下面简单的说一下各自的原理及实现方式: 一.调整视频速率 调整视频速率的原理为:修改视频的pts,dts 实现: ffmpeg -i input.m ...
- Java相关面试题总结+答案(三)
[多线程] 35. 并行和并发有什么区别? 并行:多个处理器或多核处理器同时处理多个任务.(是真正的物理上的同时发生) 并发:多个任务在同一个 CPU 核上,按细分的时间片轮流(交替)执行,从逻辑上来 ...
- ToastMiui【仿MIUI的带有动画的Toast】
版权声明:本文为HaiyuKing原创文章,转载请注明出处! 前言 仿MIUI的带有动画的Toast 效果图 代码分析 ToastMiui类基于WindowManager 为了和Toast用法保持一致 ...
- 一个请求过来都经过了什么?(Thrift版)
一.背景 最初遇到这个问题是去58面试.部门领导是原同事,所以面试比较水.水到什么程度呢? 面试就是走个形式而已,不会不过的. 一面面试官就问了一个问题:“一个请求过来都经过了什么?” 剩下的全是闲 ...
- 记一个 DataBindings遇到的坑,当ComboBox同时绑定DataSource与DataBindings的时候,绑定的元素一定要同名
下面的两个member1 一定要一样才行,不然会导致绑定异常 comboBox1.DisplayMember = "member1"; comboBox1.DataBindings ...
- SLAM+语音机器人DIY系列:(二)ROS入门——6.编写简单的service和client
摘要 ROS机器人操作系统在机器人应用领域很流行,依托代码开源和模块间协作等特性,给机器人开发者带来了很大的方便.我们的机器人“miiboo”中的大部分程序也采用ROS进行开发,所以本文就重点对ROS ...
- Python 为什么要使用描述符?
学习 Python 这么久了,说起 Python 的优雅之处,能让我脱口而出的, Descriptor(描述符)特性可以排得上号. 描述符 是Python 语言独有的特性,它不仅在应用层使用,在语言的 ...