0

0

高效解决“字符串是否包含所有长度为 K 的二进制码”问题

花韻仙語

花韻仙語

发布时间:2026-02-25 09:10:02

|

723人浏览过

|

来源于php中文网

原创

高效解决“字符串是否包含所有长度为 K 的二进制码”问题

本文详解 leetcode 1461 题的优化思路:避免暴力枚举与列表操作,改用滑动窗口 + 整数哈希 + 集合去重,在 o(n) 时间内判定二进制字符串是否包含全部长度为 k 的子码。

本文详解 leetcode 1461 题的优化思路:避免暴力枚举与列表操作,改用滑动窗口 + 整数哈希 + 集合去重,在 o(n) 时间内判定二进制字符串是否包含全部长度为 k 的子码。

在解决 LeetCode 第 1461 题「Check If a String Contains All Binary Codes of Size K」时,初学者常采用“生成全部 2ᵏ 个二进制字符串 → 逐个检查是否为子串”的暴力策略。该方法逻辑直观,但存在严重性能瓶颈:

  • 生成所有长度为 k 的二进制字符串需 O(2ᵏ) 时间与空间;
  • 每次 binary.remove(sSoFar) 或 sSoFar in binary 在列表中查找/删除均为 O(2ᵏ) 平均时间复杂度;
  • 滑动窗口中还使用了 sSoFar = sSoFar[1:] 字符串切片,每次 O(k),整体退化为 O(n·k·2ᵏ),面对 k=20(即 2²⁰ ≈ 100 万)的测试用例必然超时。

✅ 正确解法的核心优化在于 避免显式构造字符串集合,转而用整数表示二进制码,并借助哈希集合实现 O(1) 插入与去重

  1. 数学映射:每个长度为 k 的二进制子串可唯一对应一个 [0, 2ᵏ−1] 范围内的整数(如 "101" → 5);
  2. 滑动窗口整数更新:不重建子串,而是动态维护当前窗口对应的整数值:
    • 设当前值为 value,新加入位为 digit(0 或 1),则新值 = (value
    • 等价于 (value * 2 + digit) % (1
  3. 哈希集合计数:用 set() 存储已出现的整数值,一旦集合大小达到 2ᵏ,立即返回 True。

以下是优化后的 Python 实现(含关键注释):

HIX.AI
HIX.AI

HIX.AI是一个多功能的一体化AI写作助手,集成了120多种AI写作工具,支持50多种语言,能够满足各种写作需求。

下载
class Solution:
    def hasAllCodes(self, s: str, k: int) -> bool:
        # 边界判断:若字符串长度不足 k,不可能覆盖所有 k 位码
        if len(s) < k:
            return False

        total = 1 << k  # 即 2^k,共需收集的唯一二进制码数量
        seen = set()    # 存储已出现的整数型二进制码

        # 初始化:取前 k-1 位构成初始 value(补前导 0 避免 k==1 时索引错误)
        # 更简洁做法:直接从索引 0 开始,用前 k 位初始化 value
        value = 0
        # 构造初始窗口 [0, k-1] 对应的整数值
        for i in range(k):
            value = (value << 1) | int(s[i])

        seen.add(value)

        # 滑动窗口:从第 k 位开始,逐个推进
        for i in range(k, len(s)):
            # 移除最高位(第 i-k 位),加入新位 s[i]
            # 等价于:value = ((value << 1) | int(s[i])) & ((1 << k) - 1)
            value = ((value << 1) | int(s[i])) & ((1 << k) - 1)
            seen.add(value)

            # 提前终止:一旦收集满 2^k 个不同值,立刻返回
            if len(seen) == total:
                return True

        return len(seen) == total

? 关键细节说明

  • & ((1
  • 使用 | int(s[i]) 替代 + int(s[i]) 更符合位运算语义,语义清晰;
  • 提前返回机制(if len(seen) == total: return True)显著提升平均性能,尤其在答案为 True 时无需遍历完整字符串;
  • 空间复杂度降至 O(2ᵏ),时间复杂度严格为 O(n),与 k 无关(仅哈希表扩容有常数影响)。

⚠️ 注意事项

  • 切勿在循环中使用 list.remove() 或 list.pop(0) —— 它们是 O(n) 操作,会将算法拖入 O(n·2ᵏ) 泥潭;
  • 避免字符串拼接(如 sSoFar += s[j])和切片(如 sSoFar[1:]),它们在 Python 中产生新对象,开销大;
  • 当 k 较大(如 ≥ 20)时,2ᵏ 可能达百万级,此时内存虽可接受,但暴力枚举绝对不可行——本解法正是为此场景而生。

总结:本题本质是「滑动窗口 + 哈希去重」的经典应用。抓住“二进制子串 ↔ 整数映射”这一抽象,摒弃字符串操作,用位运算维持状态,辅以集合与早停策略,即可优雅通过所有测试用例,包括最大规模输入。

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

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

850

2023.08.02

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

830

2023.08.22

js 字符串转数组
js 字符串转数组

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

638

2023.08.03

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

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

217

2023.09.04

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

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

1558

2023.10.24

字符串介绍
字符串介绍

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

642

2023.11.24

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

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

1027

2024.03.22

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

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

980

2024.04.29

Golang 生态工具与框架:扩展开发能力
Golang 生态工具与框架:扩展开发能力

《Golang 生态工具与框架》系统梳理 Go 语言在实际工程中的主流工具链与框架选型思路,涵盖 Web 框架、RPC 通信、依赖管理、测试工具、代码生成与项目结构设计等内容。通过真实项目场景解析不同工具的适用边界与组合方式,帮助开发者构建高效、可维护的 Go 工程体系,并提升团队协作与交付效率。

1

2026.02.24

热门下载

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

精品课程

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

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