0

0

c++中如何实现字符串的快速匹配算法_c++ Sunday算法实现步骤【详解】

冰火之心

冰火之心

发布时间:2026-01-22 08:08:02

|

490人浏览过

|

来源于php中文网

原创

Sunday算法更快因失配时依主串模式串后字符查表跳转,非仅移1位;shift表对字符c取其在pattern最右位置i,则shift[c]=len−1−i,否则为len+1,ASCII下用vector(256,len+1)实现。

c++中如何实现字符串的快速匹配算法_c++ sunday算法实现步骤【详解】

为什么 Sunday 算法比朴素匹配快?

因为 Sunday 算法在失配时,不是只右移 1 位,而是根据主串中「模式串右侧下一个字符」的值,查预计算的偏移表,一次性跳过若干位置。平均时间复杂度是 O(n)(n 是主串长度),最坏也是 O(nm),但实际远优于朴素算法,尤其当字符集不大、模式串较短时。

关键点在于:跳得够狠,且跳得有依据——不看模式串本身失配位置,而看主串里“模式串屁股后面那个字符”是否在模式串里出现过。

如何构建 Sunday 的 shift 表?

shift 表本质是一个字符到「右移步数」的映射。对每个字符 c,定义:

  • 如果 c 在模式串 pattern 中最后一次出现在索引 i(从 0 开始),则 shift[c] = pattern.length() - 1 - i
  • 如果 c 不在 pattern 中,则 shift[c] = pattern.length() + 1(直接跳过整个模式串+1)。

注意:必须覆盖所有可能出现在主串中的字符,但实际中常只处理 ASCII(0–127)或扩展 ASCII(0–255)。用 std::vectorstd::array 最方便,初始化为默认偏移(即 pattern.size() + 1)。

立即学习C++免费学习笔记(深入)”;

std::vector buildShiftTable(const std::string& pattern) {
    std::vector shift(256, pattern.size() + 1);
    for (int i = 0; i < pattern.size(); ++i) {
        shift[static_cast(pattern[i])] = pattern.size() - i;
    }
    return shift;
}

Sunday 主循环怎么写才不出错?

核心是维护主串起始比对位置 i,每次比较从 i 开始的 pattern.size() 个字符;失配时,取 text[i + pattern.size()](即模式串右侧第一个字符),查 shift 表决定下一次 i 的增量。

知元AI
知元AI

AI智能语音聊天 对讲问答 AI绘画 AI写作 AI创作助手工具

下载

容易踩的坑:

  • i + pattern.size() 可能越界——必须保证 i 才进入比对;
  • 查表前必须把字符转成 unsigned char,否则负值索引会崩;
  • 偏移量加的是 shift[...],不是减,且要确保 i 不回退(Sunday 不回溯)。
int sundaySearch(const std::string& text, const std::string& pattern) {
    if (pattern.empty()) return 0;
    if (pattern.size() > text.size()) return -1;
auto shift = buildShiftTable(pattern);
int i = 0;
while (i <= static_cast(text.size()) - static_cast(pattern.size())) {
    bool matched = true;
    for (int j = 0; j < pattern.size(); ++j) {
        if (text[i + j] != pattern[j]) {
            matched = false;
            break;
        }
    }
    if (matched) return i;

    // 失配:看 text[i + pattern.size()] 对应的 shift 值
    if (i + pattern.size() < text.size()) {
        unsigned char c = text[i + pattern.size()];
        i += shift[c];
    } else {
        break;
    }
}
return -1;

}

和 KMP、BM 比,Sunday 适合什么场景?

Sunday 实现简单、常数小、缓存友好,特别适合:

  • 模式串较短(字节)且重复搜索不多的场景;
  • 嵌入式或教学用途——代码不到 30 行,逻辑清晰;
  • 主串为内存连续字符串(如 std::string),无随机访问瓶颈。

不适合:

  • 超长模式串(> 1KB),此时 shift 表空间开销大,且 BM 的后缀预处理更高效;
  • 需要找全部匹配位置(而不仅是首个),需微调循环但易漏边界;
  • Unicode 字符串(UTF-8)——不能直接用 char 查表,得先解码或改用 std::unordered_map,性能打折。

真正要注意的,是 buildShiftTable 中对字符类型的强制转换和主循环里那个 i + pattern.size() 的判断——少一个括号或类型错,就段错误或死循环。

相关文章

c++速学教程(入门到精通)
c++速学教程(入门到精通)

c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

258

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1468

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

621

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

551

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

566

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

166

2025.07.29

c++字符串相关教程
c++字符串相关教程

本专题整合了c++字符串相关教程,阅读专题下面的文章了解更多详细内容。

81

2025.08.07

AO3中文版入口地址大全
AO3中文版入口地址大全

本专题整合了AO3中文版入口地址大全,阅读专题下面的的文章了解更多详细内容。

1

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
CSS3 教程
CSS3 教程

共18课时 | 4.7万人学习

Sass 教程
Sass 教程

共14课时 | 0.8万人学习

Pandas 教程
Pandas 教程

共15课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号