BF 算法

目录

BF算法

算法思路

完整代码

时间复杂度

查找所有起始位置


BF算法

BF算法:即暴力(Brute Force)算法,是一种模式匹配算法,将目标串 S 的第一个字符与模式串 T 的第一个字符进行匹配,若相等,则继续比较 S 的第二个字符和 T 的第二个字符;若不相等,则比较 S 的第二个字符和 T 的第一个字符,依次比较下去,直到得出最后的匹配结果

例如:

给定字符串S :"abcdef" 作为主串,给定字符串T: "cd" 作为子串,此时,就需要在主串中查找是否出现子串,若出现,则返回主串中第一个匹配字符的下标;若未出现,则返回 -1

接下来,我们就来看 BF 算法是如何在主串中查找子串的,也就是 BF 算法的思路

算法思路

我们通过一个具体的例子来看:

定义 i 指向主串的 0 下标,j 指向子串的 0 下标

以主串的 0 下标位置作为起始位置开始匹配:

判断 i 位置的字符是否和 j 位置的字符相等, i 位置和 j 位置字符都为 a,相等,则这两个字符匹配成功,i 和 j 向后移动,继续匹配

继续判断 i 位置字符 是否与 j 位置字符相等, i 位置和 j 位置字符都为 b,相等,这两个字符也匹配成功,i 和 j 继续向后移动,进行匹配

继续判断 i 位置字符 是否与 j 位置字符相等,i 位置字符为 c,j 位置字符为 e,不相等,说明以 a(0 下标位置) 为起始位置匹配子串匹配失败

此时,需要将 i 进行回退,由于 a(0下标)作为起始位置匹配失败,因此,继续以 b (1下标)作为起始位置开始尝试匹配,也就是以 原起始位置 + 1 作为新的起始位置开始重新匹配

同样,j 也需要回退,需要将 j 回退到 0 下标位置,与下一个起始位置重新开始匹配

 由于 i 向前移动,当匹配失败时,我们该如何确定起始位置呢?

 由于当 i 和 j 匹配成功时,两者会同时向后移动,因此,通过 j 就可以知道 i 向前移动了多少步

 i - j 就是本次匹配的起始位置,而 i - j + 1 也就是新的起始位置

以 1 下标为起始位置继续匹配:

 i 位置字符为 b,j 位置字符为 a,不相等,说明以 b 为起始位置匹配子串匹配失败,再次回退,由于 i = 1,j = 0,因此,新的起始位置为 2,j 的位置为 0 

继续匹配,  i 位置字符为 c,j 位置字符为 a,不相等,说明以 c 为起始位置匹配子串匹配失败,再次回退,由于 i = 2,j = 0,因此,新的起始位置为 3,j 的位置为 0 

继续匹配, i 位置和 j 位置字符都为 a,相等,这两个字符匹配成功,i 和 j 向后移动

继续匹配, i 位置和 j 位置字符都为 b,相等,这两个字符也匹配成功,i 和 j 向后移动

 继续匹配, i 位置和 j 位置字符都为 e,相等,这两个字符也匹配成功,i 和 j 向后移动 

此时,子串已经遍历完了,也就说明 子串 中的所有字符都与 主串 中的字符相匹配,在主串中找到了子串,此时,就可以直接返回主串中本次匹配的起始位置,也就是 i - j

而若是主串先遍历完成,也就说明 主串 中没有与 子串 中字符完全匹配的字符串,找不到,此时,就需要返回 -1

总结一下上述过程:

1. 定义 i 指向主串的 0 下标,j 指向子串的 0 下标

2. 判断 i 位置字符与 j 位置字符是否相同,若相同,i 和 j 同时向后移动(i++ 且 j++);若不同,则i 和 j 都需要进行回退(i = i - j + 1,j = 0)

3. 重复 2 过程,直到遍历完 主串 或 子串,若子串先遍历完,说明在主串中找到了子串,匹配成功,返回起始位置(i - j);若主串先遍历完,说明主串中找不到子串,匹配失败,返回 -1

在分析了 BF 算法的思路之后,我们就来尝试编写代码

完整代码

    /*** 判断主串中是否含有子串* @param str 主串* @param target 子串* @return 主串中含有子串,返回主串中子串第一次出现的起始下标;若不含有,返回 -1*/public int BF(String str, String target) {// 处理特殊情况if (str == null || target == null) {return -1;}int strLen = str.length();int targetLen = target.length();if (strLen == 0 || strLen < targetLen) {return -1;}// 开始判断int i = 0, j = 0;while (i < strLen && j < targetLen) {// 判断 str 中 i 位置字符是否与 target 中j 位置字符相同if (str.charAt(i) == target.charAt(j)) {// 相同,i 和 j 都向后移动i++;j++;} else {// 不相同,i 和 j 都进行回退i = i - j + 1;j = 0;}}// 子串先遍历完if (j >= targetLen) {return i - j;} else {// 主串先遍历完return -1;}}

接下来,我们来分析 BF 算法的时间复杂度

时间复杂度

假设主串长度为 M,子串长度为 N

在最坏情况下,

在主串的前 M - N 个位置,每一次匹配时,都要比较到子串的最后一个字符(比较 N 次),才发现匹配不成功,然后再将 i 和 j 退回,继续比较,一共比较了 (M - N) * N

在 主串的最后 N 个位置,也分别匹配了 N、N - 1、N - 2...次,也就是 (N + 1)(N) / 2

因此,时间复杂度为 O(M*N)

上述我们查找的是主串中子串第一次出现时的起始位置,那么, 若我们想要查找主串中与子串相同的字符串的所有起始位置,该如何实现呢?

查找所有起始位置

与查找 主串中子串第一次出现的起始位置 思路相同

但是,当我们找到第一个起始位置(也就是 j 第一次遍历完子串时)时,不能直接返回,而是要继续在主串中查找

例如:

j 第一次遍历完子串:

以 0 下标为起始的字符串与子串匹配成功,将 0 下标存储起来,然后继续查找 

此时,以 i + 1 为起始位置的字符串可能与字符相同,因此需要将 i 进行回退,继续查找

同时,也需要将 j 回退到 0 下标位置,重新开始匹配

 

总结一下上述过程:

1. 定义 i 指向主串的 0 下标,j 指向子串的 0 下标

2. 判断 i 位置字符与 j 位置字符是否相同,若相同,i 和 j 同时向后移动(i++ 且 j++);若不同,则i 和 j 都需要进行回退(i = i - j + 1,j = 0)

3. 判断 j 是否遍历完子串,若遍历完,则存储起始位置(i - j),再将 i 和 j 进行回退(i = i - j + 1,j = 0)

4. 重复 2、3 直到主串遍历完成

 代码实现:

    /*** 查找主串中与子串相同的字符串的所有起始位置* @param str 主串* @param target 子串* @return 查询结果集*/public List<Integer> BF(String str, String target) {List<Integer> ret = new ArrayList<>();// 处理特殊情况if (str == null || target == null) {return ret;}int strLen = str.length();int targetLen = target.length();if (strLen == 0 || strLen < targetLen) {return ret;}// 开始判断int i = 0, j = 0;while (i < strLen) {// 判断 str 中 i 位置字符是否与 target 中j 位置字符相同if (str.charAt(i) == target.charAt(j)) {// 相同,i 和 j 都向后移动i++;j++;} else {// 不相同,i 和 j 都进行回退i = i - j + 1;j = 0;}// 判断子串是否遍历完成if (j >= targetLen) {ret.add(i - j);i = i - j + 1;j = 0;}}return ret;}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/55746.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【最新华为OD机试E卷-支持在线评测】TLV解码(100分)多语言题解-(Python/C/JavaScript/Java/Cpp)

🍭 大家好这里是春秋招笔试突围 ,一枚热爱算法的程序员 💻 ACM金牌🏅️团队 | 大厂实习经历 | 多年算法竞赛经历 ✨ 本系列打算持续跟新华为OD-E/D卷的多语言AC题解 🧩 大部分包含 Python / C / Javascript / Java / Cpp 多语言代码 👏 感谢大家的订阅➕ 和 喜欢�…

【git】如何快速准确的回退(reverse)已经合并(merge)主分支(master)的新提交代码

文章目录 前言一、merge模式二、回滚步骤总结 前言 我们在做一些需求&#xff0c;正常流程经过开发&#xff0c;测试到最后和代码上线。但是有时候就会发生一些小插曲&#xff0c;比如产品说老板说某某某你的代码要延后上线&#xff01;&#xff01;或者你写的不合格预发环境出…

在Openshift(K8S)上通过EMQX Operator部署Emqx集群

EMQX Operator 简介 EMQX Broker/Enterprise 是一个云原生的 MQTT 消息中间件。 我们提供了 EMQX Kubernetes Operator 来帮助您在 Kubernetes 的环境上快速创建和管理 EMQX Broker/Enterprise 集群。 它可以大大简化部署和管理 EMQX 集群的流程&#xff0c;对于管理和配置的知…

ubuntu 安装keepalived+haproxy

一、安装keepalived sudo apt update sudo apt install keepalived sudo systemctl start keepalived sudo systemctl enable keepalived sudo systemctl status keepalived#配置Keepalived sudo cp /etc/keepalived/keepalived.conf.sample /etc/keepalived/keepalived.conf …

Java面试宝典-并发编程学习02

目录 21、并行与并发有什么区别&#xff1f; 22、多线程中的上下文切换指的是什么&#xff1f; 23、Java 中用到的线程调度算法是什么&#xff1f; 24、Java中线程调度器和时间分片指的是什么&#xff1f; 25、什么是原子操作&#xff1f;Java中有哪些原子类&#xff1f; 26、w…

Python案例小练习——小计算器

文章目录 前言一、代码展示二、运行展示 前言 这是用python实现一个简单的计器。 一、代码展示 def calculate(num1, op, num2):if op "":return float(num1) float(num2)elif op "-":return float(num1) - float(num2)elif op "*":return…

【Mac苹果电脑安装】DBeaverEE for Mac 数据库管理工具软件教程【保姆级教程】

Mac分享吧 文章目录 DBeaverEE 数据库管理工具 软件安装完成&#xff0c;打开效果图片Mac电脑 DBeaverEE 数据库管理工具 软件安装——v24.21️⃣&#xff1a;下载软件2️⃣&#xff1a;安装JDK&#xff0c;根据下图操作步骤提示完成安装3️⃣&#xff1a;安装DBeaverEE&#…

C++类域访问方式(public,protected,private)对象访问 , 通过成员函数访问 ,通过友元函数访问

c类的用法 yC 类的基础用法与详细说明&#xff1a;简单易懂的入门指南-CSDN博客 类的基本概念&#x1f447; 类是C中的一个用户定义的数据类型&#xff0c;它可以包含数据&#xff08;成员变量&#xff09;和函数&#xff08;成员函数&#xff09;。通过类&#xff0c;我们可以…

【Jenkins】windows安装步骤

【Jenkins】windows安装步骤 官网使用WAR包方式运行浏览器访问Jenkinswindows-installer安装安装过程问题解决This account either does not hava the privilege to logon as a service or the account was unable to be verified 安装成功修改jenkins.xml启动jenkins访问jenki…

springboot030甘肃非物质文化网站的设计与开发(论文+源码)_kaic

毕 业 设 计&#xff08;论 文&#xff09; 题目&#xff1a;甘肃非物质文化网站设计与实现 摘 要 现代经济快节奏发展以及不断完善升级的信息化技术&#xff0c;让传统数据信息的管理升级为软件存储&#xff0c;归纳&#xff0c;集中处理数据信息的管理方式。本甘肃非物质文化…

Java基于SSM框架的教学辅助微信小程序【附源码、文档】

博主介绍&#xff1a;✌IT徐师兄、7年大厂程序员经历。全网粉丝15W、csdn博客专家、掘金/华为云//InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专栏推荐订阅&#x1f447;&#x1f3…

vim实用笔记

函数跳转功能 想要使用函数跳转功能需要先安装 ctags sudo apt-get install exuberant-ctags接着&#xff0c;在源文件目录树执行如下命令&#xff1a; ctags -R . 即可在该目录下生成一个tags文件&#xff0c; 这个文件就是所有函数和变量的索引 接着打开用vim打开任一文件…

BinAbsInspector部署、使用与原理分析

文章目录 前言1、概述2、安装与使用2.1、源码安装2.1.1、部署系统依赖组件2.1.1.1、部署基础依赖组件2.1.1.2、部署Ghidra 11.0.32.1.1.2.1、部署JDK 172.1.1.2.2、部署Gradle 7.4.22.1.1.2.3、部署Ghidra 11.0.3 2.1.1.3、部署Z3 4.8.15 2.1.2、使用源码安装系统 2.2、使用方法…

CMake 教程(二)添加库

目录 一、实例一——创建库1、add_library2、target_include_directories()、target_link_libraries()2.1 target_include_directories()2.2 target_link_libraries() 3、实例操作 二、实例二——添加选项1、option()2、实例操作 在第一节 CMake 教程&#xff08;一&#xff09…

学习笔记(202410)

课程&#xff1a;Generative AI for Software Development 链接&#xff1a;吴恩达同步最新AI专业课&#xff0c;第54讲&#xff1a;用人工智能做软件开发--Generative AI for Software Development_哔哩哔哩_bilibili 时间&#xff1a;2024-10-12 至 概述&#xff1a;使用C…

ES-入门-javaApi-文档-新增-删除

新增指定索引的文档数据的代码如下&#xff1a; package com.atgulgu.es.test;import com.fasterxml.jackson.databind.ObjectMapper; import org.apache.http.HttpHost; import org.elasticsearch.action.index.IndexRequest; import org.elasticsearch.action.index.IndexRe…

学习笔记——交换——STP(生成树)简介

一、技术背景 1、生成树技术背景 交换机单线路组网&#xff0c;存在单点故障(上左图)&#xff0c;上行线路及设备都不具备冗余性&#xff0c;一旦链路或上行设备发生故障&#xff0c;业务将会中断。 为了使得网络更加健壮、更具有冗余性&#xff0c;将拓扑修改为(上右图)接入…

网络学习笔记

一、网络的结构与功能 网络的鲁棒性与抗毁性 如果在移走少量节点后网络中的绝大部分节点仍然是连通的&#xff0c;那么就该网络的连通性对节点故障具有鲁棒性 网络上的动力学 动力系统&#xff1a;自旋、振子或混沌的同步、可激发系统 传播过程&#xff1a;信息传播与拥堵…

ChatGPT01-preivew体验报告:内置思维链和多个llm组合出的COT有啥区别呢?丹田与练气+中学生物理奥赛题测试,名不虚传还是名副其实?

一个月前&#xff0c;o1发布的时候&#xff0c;我写了篇文章介绍 逻辑推理能力堪比博士生&#xff0c;OpenAI发布全新AI模型系列&#xff1a; o1 - 大模型或许进入新阶段&#xff0c;还翻译了官方的介绍 解密OpenAI o1是如何让LLMs获得逻辑推理能力的 - CoT * RL&#xff0c;也…

代码复现(五):GCPANet

文章目录 net.py1.class Bottleneck&#xff1a;残差块2.class ResNet&#xff1a;特征提取3.class SRM&#xff1a;SR模块4.class FAM&#xff1a;FIA模块5.class CA&#xff1a;GCF模块6.class SA&#xff1a;HA模块7.class GCPANet&#xff1a;网络架构 train.pytest.py 论文…