最长前缀数组

欢迎关注更多精彩
关注我,学习常用算法与数据结构,一题多解,降维打击。

基本问题

给定一个字符串s, 返回一个数组Z, Z[i]代表子串s[i…n] 与s最长公共前缀的长度。
在这里插入图片描述

朴素做法

可以枚举所有s[i…n]子串,然后与s一一比较,复杂度O(n*n)

优化

可以参照KMP的思想,利用已经判断过的相同串。

在这里插入图片描述

利用该思想可以将复杂度缩小到O(n)


vector<int> Zfunc(string& str) {int n = str.size();vector<int>z(n);int l = 0, r = 0;for (int i = 1; i < n; i++) {if (i <= r) {z[i] = min(r - i + 1, z[i - l]);}// 循环只有在i + z[i]>r时才会运算。总运算次数为n次。while (i + z[i] < n && str[z[i]] == str[i + z[i]]) {z[i]++;}if (i + z[i] - 1 > r) {l = i;r = i + z[i] - 1;}}return z;
}

题目一

https://codeforces.com/contest/1968/problem/G1

题目大意

给定一个字符串s和数字k,问将字符串分成连续k段,公共前缀最长是多少。

题目分析

可以通过二分枚举长度l,然后查找s[0…l]在s中出现的次数,利用公共前缀数组实现每次判断为O(n), 整体复杂度为nlog(n)。

代码

#include <iostream>
#include <stdio.h>
#include <queue>
#include <string.h>
#include <stack>
#include <vector>
#include <map>
#include <algorithm>
#include <assert.h>using namespace std;typedef long long lld;vector<int> Zfunc(string & str) {int n = str.size();vector<int>z(n);int l = 0, r = 0;for (int i = 1; i < n; i++) {if (i <= r) {z[i] = min(r - i + 1, z[i - l]);}while (i + z[i] < n && str[z[i]] == str[i + z[i]]) {z[i]++;}if (i + z[i] - 1 > r) {l = i;r = i + z[i] - 1;}}return z;
}int getCnt(vector<int> Z, int len) {int cnt = 1;for (int i = len; i < Z.size();) {if (Z[i] >= len) {cnt++;i += len;}else i++;}return cnt;
}void solve() {int t;cin >> t;while (t--) {int n, l, r;cin >> n >> l >> r;string s;cin >> s;auto Z = Zfunc(s);/*for (int z : Z)cout << z << " ";cout << endl;*/int left = 0, right = s.length() / l;while (left < right) {int mid = (left + right) / 2 + (left + right) % 2;if (getCnt(Z, mid) >= r) {left = mid;}else {right = mid - 1;}}cout << left << endl;}
}int main() {solve();return 0;
}
/*
7
3 3 3
aba
3 3 3
aaa
7 2 2
abacaba
9 4 4
abababcab
10 1 1
codeforces
9 3 3
abafababa
5 3 3
zpozp2
10 1 1
aaaaaaaaaa
10 1 1
abcdaabcda*/

题目二

https://codeforces.com/contest/1968/problem/G2

题目大意

该题是上一题的加强版本。
给定一个字符串s和数字l,r,问将字符串分成连续k段(k=l,l+1,l+2,…,r),公共前缀最长是多少。

题目分析

先求出k=1到s.length()的所有答案,用数组ans表示。
可以分段考虑,对于k<=sqrt(n), 可以按照题目一的做法,总复杂度为sqrt(n) n log(n)
对于k>sqrt(n), 那么分段后的长度不会大于sqrt(n), 可以枚举l 从1到sqrt(n) 计算出最多可以分成多少段k,则ans[k]=l。
保险起见,最后做一遍最大值比较,ans[i-1]=max(ans[i-1], ans[i])。

代码


#include <iostream>
#include <stdio.h>
#include <queue>
#include <string.h>
#include <stack>
#include <vector>
#include <map>
#include <algorithm>
#include <cmath>using namespace std;typedef long long lld;vector<int> Zfunc(string & str) {int n = str.size();vector<int>z(n);int l = 0, r = 0;for (int i = 1; i < n; i++) {if (i <= r) {z[i] = min(r - i + 1, z[i - l]);}while (i + z[i] < n && str[z[i]] == str[i + z[i]]) {z[i]++;}if (i + z[i] - 1 > r) {l = i;r = i + z[i] - 1;}}return z;
}int getCnt(vector<int>& Z, int len) {int cnt = 1;for (int i = len; i < Z.size();) {if (Z[i] >= len) {cnt++;i += len;}else i++;}return cnt;
}int getFix(vector<int >& Z, string &s, int k) {int left = 0, right = s.length() / k;while (left < right) {int mid = (left + right) / 2 + (left + right) % 2;if (getCnt(Z, mid) >= k) {left = mid;}else {right = mid - 1;}}return left;
}void solve() {int t;cin >> t;while (t--) {int n, l, r;cin >> n >> l >> r;string s;cin >> s;auto Z = Zfunc(s);vector<int> ans(s.length() + 1, 0);int sl = sqrt(s.length()) + 0.5;for (int i = 1; i <= sl; ++i) {ans[i] = max(ans[i], getFix(Z, s, i));int k = getCnt(Z, i);ans[k] = max(ans[k], i);}for (int i = s.length()-1; i > 0; --i) {ans[i] = max(ans[i + 1], ans[i]);}for (; l <= r; ++l) {cout << ans[l] << " ";}cout << endl;}
}int main() {solve();return 0;
}
/*
7
3 3 3
aba
3 3 3
aaa
7 2 2
abacaba
9 4 4
abababcab
10 1 1
codeforces
9 3 3
abafababa
5 3 3
zpozp2
10 1 1
aaaaaaaaaa
10 1 1
abcdaabcda7
3 1 3
aba
3 2 3
aaa
7 1 5
abacaba
9 1 6
abababcab
10 1 10
aaaaaaawac
9 1 9
abafababa
7 2 7
vvzvvvv*/

本人码农,希望通过自己的分享,让大家更容易学懂计算机知识。创作不易,帮忙点击公众号的链接。

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

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

相关文章

ERROR TypeError: Cannot read property ‘tapPromise‘ of undefined

问题概要 vue项目运行npm run build打包时&#xff0c;报了ERROR TypeError: Cannot read property ‘tapPromise’ of undefined TypeError: Cannot read property ‘tapPromise’ of undefined的错误。 解决方式 根据github中compression-webpack-plugin源码issues里的提示 …

C#--SVG矢量图画法示例

1.代码示例 <Viewbox Grid.Column"1" Grid.ColumnSpan"1" Grid.RowSpan"1" ><Path Name"ValveShape" Stroke"Black" Data"M 50,0 L 150,200 L 50,200 L 150,0 Z" Width"200" Height"…

经常出差用哪些办公软件记录工作?可多设备同步使用的便签笔记软件

对于许多职场人士来说&#xff0c;出差已成为工作常态。在旅途中&#xff0c;如何高效处理工作&#xff0c;确保信息不遗漏&#xff0c;成为了一个不小的挑战。那么&#xff0c;对于经常需要移动办公的我们&#xff0c;哪款办公软件才是最佳选择呢&#xff1f;可多设备同步使用…

5.23 学习总结

一.项目优化&#xff08;语音通话&#xff09; 实现步骤&#xff1a; 1.用户发送通话申请&#xff0c;并处理通话请求&#xff0c;如果同意&#xff0c;为两个用户之间进行连接。 2.获取到电脑的麦克风和扬声器&#xff0c;将获取到的语音信息转换成以字节数组的形式传递。 …

DataBinding viewBinding(视图绑定与数据双向绑定)简单案例 (kotlin)

先上效果&#xff1a; 4个view的文字都是通过DataBinding填充的。交互事件&#xff1a;点击图片&#xff0c;切换图片 创建项目&#xff08;android Studio 2023.3.1&#xff09; Build.gradle(:app) 引入依赖库&#xff08;完整源码&#xff09; buildFeatures { vie…

java中的HashSet类

一、HashSet类 实现了Set接口&#xff0c;无法存储重复元素 特点&#xff1a;元素位置无序、无索引、底层是HashMap 1、构造方法 内部是HashMap的构造方法 2、add方法 (1)元素在底层存储使用到了三种数据结构&#xff1a;hash数组、链表、树 (2)添加流程&#xff08;根据…

告别登录烦恼,WPS免登录修改器体验!(如何实现不登录使用WPS)

文章目录 &#x1f4d6; 介绍 &#x1f4d6;&#x1f3e1; 演示环境 &#x1f3e1;&#x1f4d2; 解决方案 &#x1f4d2;&#x1f388; 获取方式 &#x1f388;⚓️ 相关链接 ⚓️ &#x1f4d6; 介绍 &#x1f4d6; 想象一下&#xff0c;如果你能够绕过繁琐的登录流程&#x…

C# WinForm —— 22 Forms.Timer 组件介绍与使用

1. 简介 Timer 计时器 组件不会出现在窗体中&#xff0c;每隔 定义的间隔时间&#xff0c;就会触发事件&#xff0c;主要应用于Windows应用程序&#xff0c;独占一个线程&#xff0c;可以修改 UI 元素 属性 属性解释(Name)控件ID&#xff0c;在代码里引用的时候会用到,一般…

运营美区TikTok小店常见问题汇总,你中了几个?

大家好&#xff0c;我是IPdodo的小编&#xff0c;专注于分享出海网络解决方案&#xff0c;致力于为TikTok运营人提供解决视频0播放、直播间卡顿、不进人甚至封号等问题的跨境网络专线。目前已经帮助数千位用户成功开启跨境业务。 今天&#xff0c;将针对美区TikTok小店的常见问…

Swift使用JSONDecoder处理json数据,实现json序列化和反序列化

Json数据处理是开发中不可获取的一项技能&#xff0c;如果你不会处理json数据&#xff0c;那你离失业就不远了&#xff0c;所以学完了swift基础教程&#xff0c;还是先老老实实学习一下json处理吧&#xff0c;有了这项技能&#xff0c;你才可以继续下一个网络请求阶段的开发&am…

Git总结超全版

最近想系统的回顾一下Git的使用&#xff0c;如果只想快速的集成git到idea&#xff0c;可以参考另一篇我的博客中的git部分 目录 版本管理工具简介Git安装与配置Git远程仓库配置 Git常用命令为常用命令配置别名(可选)Git忽略文件.gitignore一些概念*本地仓库操作删除仓库内容 *远…

滴滴三面 | Go后端研发

狠狠的被鞭打了快两个小时… 注意我写的题解不一定是对的&#xff0c;如果你认为有其他答案欢迎评论区留言 bg&#xff1a;23届 211本 社招 1. 自我介绍 2. 讲一个项目的点&#xff0c;因为用到了中间件平台的数据同步&#xff0c;于是开始鞭打数据同步。。 3. 如果同步的时候…

Sketch v100 for Mac 安装教程【支持M芯片】

Sketch v100 for Mac 安装教程【支持M芯片】 原文地址&#xff1a;https://blog.csdn.net/weixin_48311847/article/details/139104315

CasaOS系统玩客云安装内网穿透工具实现无公网IP远程访问

文章目录 前言1. CasaOS系统介绍2. 内网穿透安装3. 创建远程连接公网地址4. 创建固定公网地址远程访问 前言 2月底&#xff0c;玩客云APP正式停止运营&#xff0c;不再提供上传、云添加功能。3月初&#xff0c;有用户进行了测试&#xff0c;局域网内的各种服务还能继续使用&am…

【学习笔记】后端(Ⅰ)—— NodeJS(Ⅰ)

NodeJS 1、概述 1.1、NodeJS是什么 1.2、NodeJS的主要作用 1.3、NodeJS的优点 1.4、NodeJS 与 浏览器 的 JavaScript 对比 1.4.1 ECMAScript 介绍 1.4.2 JavaScript 介绍 1.4.3 TypeScript 介绍2、基础篇 2.1、Buff…

LangChain带你轻松玩转ChatGPT等大模型开发

大家好&#xff0c;我是herosunly。985院校硕士毕业&#xff0c;现担任算法研究员一职&#xff0c;热衷于机器学习算法研究与应用。曾获得阿里云天池比赛第一名&#xff0c;CCF比赛第二名&#xff0c;科大讯飞比赛第三名。拥有多项发明专利。对机器学习和深度学习拥有自己独到的…

Python中cv2 (OpenCV, opencv-python)库的安装、使用方法demo最新详细教程

&#x1f42f; Python中cv2 (OpenCV, opencv-python)库的安装、使用方法demo最新详细教程 &#x1f4f8; 文章目录 &#x1f42f; Python中cv2 (OpenCV, opencv-python)库的安装、使用方法demo最新详细教程 &#x1f4f8;摘要引言正文&#x1f4d8; OpenCV库概述&#x1f680; …

【Git教程】(十九)合并小型项目 — 概述及使用要求,执行过程及其实现,替代解决方案 ~

Git教程 合并小型项目 1️⃣ 概述2️⃣ 使用要求3️⃣ 执行过程及其实现 在项目的初始阶段&#xff0c;往往需要针对重要的设计决策和技术实现原型实验。当原型评估结束后&#xff0c;需要将那些成功的原型合并起来称为整个项目的初始版本。 在这样的情景中&#xff0c;各个原…

php代码审计参考

代码审计思路&#xff1a; 从个人角度出发&#xff0c;如果环境允许的话&#xff0c;可以先选择做一个”程序员“再来做代码审计。因为从开发者的位置去思考问题&#xff0c;可以快速定位问题。学习面向对象编程以及面向过程编程&#xff0c;编写一些 项目提升对代码的理解能力…

Android Compose 六:常用组件 Button

Button 1 简单使用 Button(onClick { /*TODO*/ }) {Text(text "我是一只button里的text")}效果 颜色为什么是这个样子&#xff1f; 前面Text里我们讲过 主题色会影响组件的颜色 这里我使用的颜色如下 primary Color(0xFFFF0000),onPrimary Color(0xFF00FF00),p…