B-Suffix Array

B-Suffix Array

题意:

一个字符串只含有a和b,先给出b数组的构造方式:
对于每个位置i来说:

  1. 如果存在一个位置j,使得j<i,且s[j] == s[i],则b[i]=i-j
  2. 否则b[i]=0
    现在对字符串每个后缀都构造B数组,并按照字典序排序

题解:

参考博客
题目标题就已经透露了一切,Suffix Array,说明这个题要用后缀数组来做,但是具体怎么做呢?
我们会发现,化为B数组的值后,第一个a的值肯定为0,第一个b的值肯定也为0,那么从第一个a到第一个b之间肯定都是1
比如“aaaabaaab”,则B(t) = 011102114,前半段为01110(也就是第一个a到第一个b),我们可以将B数组分为两部分,前半部为01110,后半部分为2114,这样字典序排序时,我们先对前半部分排序即可,如果前半部分一样再对后半部分排序
对于前半部分排序,直接比较长度即可,因为都是0开头,0结尾,中间是1,越长说明中间的1越多,且前半部分极好确定,直接找第一个a与b就行
那现在后半部分怎么确定呢?
Ai表示前半部分,DI表示后半部分
后半部分是数组B的后缀,我们直接后缀数组,得到rank值,rank[i+|Ai|]就是Di的排名
Rank[i]=后缀suf(i)的排名,也就是从第i位到最后一位构成的子串排名
这个题,秒极了
在这里插入图片描述

代码:

#include <bits/stdc++.h>
using namespace std;
struct _IO{_IO(){ios::sync_with_stdio(0);cin.tie(0);}}_io;
typedef long long ll; typedef long double db;
const int N = 2e6 + 5, M = 1e9 + 7;int sa[N], rk[N], oldrk[N << 1], id[N], px[N], cnt[N];
// px[i] = rk[id[i]](用于排序的数组所以叫 px)bool cmp(int x, int y, int w) {return oldrk[x] == oldrk[y] && oldrk[x + w] == oldrk[y + w];
}void da(int *s, int n, int m) {int i,p,w;for(int i=0;i<=n;i++)cnt[i]=0;for (i = 1; i <= n; ++i) ++cnt[rk[i] = s[i]];for (i = 1; i <= m; ++i) cnt[i] += cnt[i - 1];for (i = n; i >= 1; --i) sa[cnt[rk[i]]--] = i;for (w = 1; w < n; w <<= 1, m = p) {  // m=p 就是优化计数排序值域for (p = 0, i = n; i > n - w; --i) id[++p] = i;for (i = 1; i <= n; ++i)if (sa[i] > w) id[++p] = sa[i] - w;//memset(cnt, 0, sizeof(cnt));for(int i=0;i<=n;i++)cnt[i]=0;for (i = 1; i <= n; ++i) ++cnt[px[i] = rk[id[i]]];for (i = 1; i <= m; ++i) cnt[i] += cnt[i - 1];for (i = n; i >= 1; --i) sa[cnt[px[i]]--] = id[i];for(int i=0;i<=n;i++)oldrk[i]=rk[i];//memcpy(oldrk, rk, sizeof(rk));for (p = 0, i = 1; i <= n; ++i)rk[sa[i]] = cmp(sa[i], sa[i - 1], w) ? p : ++p;}}int n;
struct node {int x, y;bool operator < (const node &b) const {if (y - x == b.y - b.x) {return rk[y+1] < rk[b.y+1];}return y - x < b.y - b.x;}
} a[N];
int b[N];
char s[N];
int main() {while (cin >> n) {cin >> s+1;int x = -1, y = -1;for (int i = 1; i <= n; i++) {b[i] = 0;if (s[i] == 'a') {if (x != -1) b[i] = i - x;x = i;} else {if (y != -1) b[i] = i - y;y = i;}}for(int i=1;i<=n;i++){b[i]++;}da(b, n, n);//后缀数组求出rank数组for(int i=1;i<=n;i++)x = y = n+1;for (int i = n ; i >= 1; i--) {if (s[i] == 'a') {//从第i位开始的后缀,分为前部分Ai,范围[x,y]和后部分Bi a[i] = {i, y};x = i;} else {a[i] = {i, x};y = i;}}rk[n+1] = -1;rk[n+2] = -2;sort(a+1, a + n+1);for (int i = 1; i <=n; i++) {cout << a[i].x  << ' ';}cout << '\n';}
}

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

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

相关文章

Wannafly挑战赛10F-小H和遗迹【Trie,树状数组】

正题 题目链接:https://ac.nowcoder.com/acm/contest/72/F 题目大意 nnn个字符串&#xff0c;包括小写字母和#\##。其中#\##可以替换为任意字符串。求有多少对字符串可能相同。 保证每个字符串至少有一个#\##。 2≤n≤500000,∑i1n∣si∣≤1062\leq n\leq 500000,\sum_{i1}^n…

线段树合并、分裂

基本概念&#xff1a; 如果需要维护许多个大小为 \(10^5\) 级别的多重集&#xff0c;可以看做给每一个多重集建立一棵线段树。线段树的合并、分裂就是多重集的累加、分开。 这里使用动态开点的方式存储线段树树。 如果一个节点为空&#xff0c;那么它的编号为 \(0\) 。 变量释义…

(神奇的)虚树(初步了解)

参考文献&#xff1a; 博文1 博文2 博文3 引入 在一类树上动态规划问题中,题目给出的询问往往包含树上的很多各节点,并保证总的点数规模小于某个值. 如果我们直接在整颗树上进行dp的话,时间复杂度与询问的次数有关,这显然是不可接受的,如果我们可以找到一种动态规划的方法,使…

P2048 [NOI2010] 超级钢琴(RMQ 贪心)

文章目录题目描述解析代码传送门题目描述 解析 首先&#xff0c;如果只有一个和弦&#xff0c;那么问题显然简单了 用前缀和结合ST表随便做做即可 然而 这次要求前k大的 怎么办呢&#xff1f; 参照之前有一道序列合并的做法 我们想到&#xff0c;可以先建一个优先队列&#xf…

微服务架构基础之Service Mesh

ServiceMesh(服务网格) 概念在社区里头非常火&#xff0c;有人提出 2018 年是 ServiceMesh 年&#xff0c;还有人提出 ServiceMesh 是下一代的微服务架构基础。那么到底什么是 ServiceMesh&#xff1f;它的诞生是为了解决什么问题&#xff1f;企业是否适合引入 ServiceMesh&…

[SDOI2011]消耗战

[SDOI2011]消耗战 题意&#xff1a; 给出n个点的一棵带有边权的树,以及q个询问.每次询问给出k个点,询问这使得这k个点与1点不连通所需切断的边的边权和最小是多少. 题解&#xff1a; 树型dp虚树 dp[x]:切断x及其子树上询问点的最小代价 预处理出minv[pos]代表从11到pos路径…

【做题记录】CF1428E Carrots for Rabbits—堆的妙用

CF1428E Carrots for Rabbits 题意&#xff1a; 有 \(n\) 个萝卜&#xff0c;每个萝卜的初始大小为 \(a_i\) 。现在要把这些萝卜切为为 \(k\) 个。吃每一个萝卜的时间为这个萝卜的大小的平方&#xff0c;求吃完所有萝卜的最小时间&#xff0c;即 \(\sum_{i1}^{k}{a_i^2}\) 最小…

P4716-[模板]最小树形图

正题 题目链接:https://www.luogu.com.cn/problem/P4716 题目大意 给出nnn个点mmm条边的一张有向图&#xff0c;求以rrr为根的最小外向树。 1≤n≤100,1≤m≤1041\leq n\leq 100,1\leq m\leq 10^41≤n≤100,1≤m≤104 解题思路 考虑一种贪心&#xff0c;对于每个点我们先选出…

P1081 [NOIP2012 提高组] 开车旅行(倍增)(动态规划)

洛谷传送门 文章目录题目描述解析代码题目描述 解析 利用倍增&#xff0c;设计dp慢慢敲即可。。。 注意距离累加在一起会爆int&#xff0c;需要ll 特判条件非常之复杂。。。 心力交瘁&#xff0c;就酱了 代码 #include <bits/stdc.h> using namespace std; #define ll…

dotnet core调试docker下生成的dump文件

最近公司预生产环境.net core应用的docker容器经常出现内存暴涨现象&#xff0c;有时会突然吃掉几个G,触发监控预警&#xff0c;造成容器重启。分析了各种可能原因&#xff0c;修复了可能发生的内存泄露&#xff0c;经测试本地正常&#xff0c;但是发到预生产还是会有内存暴涨现…

【做题记录】区间排序—线段树

1. CF558E A Simple Task 题意&#xff1a; 给定由小写字母组成的字符串 \(s\) 每一次操作如下&#xff1a; \(opt0\) &#xff1a;将 \([l,r]\) 降序排序 \(opt1\) &#xff1a;将 \([l,r]\) 升序排序 输出最终字符串 题解&#xff1a; 大致思想为&#xff0c;建 \(26\) 棵线…

Quadratic Form

Quadratic Form 题意&#xff1a; 一个n * n 的正定矩阵和一个n维的向量b&#xff0c;现在找一个x1&#xff0c;x2&#xff0c;…xn满足以下条件&#xff1a; 求这个式子&#xff0c;最后输出P * Q-1 mod 998244353. 题解&#xff1a; 参考 线性代数学过n阶正定的实矩阵等…

P3426-[POI2005]SZA-Template【KMP】

正题 题目链接:https://www.luogu.com.cn/problem/P3426 题目大意 给出一个长度为nnn的字符串sss&#xff0c;求一个长度最小的字符串ttt使得sss所有ttt和ttt匹配的位置能覆盖串sss。 1≤n≤51051\leq n\leq 5\times 10^51≤n≤5105 解题思路 首先答案肯定是原串的一个borde…

8分钟学会Consul集群搭建及微服务概念

Consul介绍&#xff1a;Consul 是由 HashiCorp 公司推出的开源软件&#xff0c;用于实现分布式系统的服务发现与配置。与其他分布式服务注册与发现的方案&#xff0c;Consul 的方案更“一站式”&#xff0c;内置了服务注册与发现框 架、分布一致性协议实现、健康检查、Key/Valu…

取模除法(逆元)(费马小定理)(线性求逆元)

文章目录引言逆元费马小定理内容应用证明线性求逆元thanks for reading&#xff01;引言 我们做题时经常会由于答案过大&#xff0c;被要求使答案对一个质数取模 我们都知道&#xff0c;加和乘对取模是没有影响的 减法也只需要写一个&#xff1a; int mod_minus(int a,int b)…

数学基础知识(高精、快速幂、龟速乘……)

压位高精 模板代码 龟速乘 ll ch(ll x,ll y) {ll ret0;while(y){if(y&1) ret(retx)%m;x(xx)%m,y>>1;}return ret; }

P2742 [USACO5.1]圈奶牛Fencing the Cows /【模板】二维凸包

P2742 [USACO5.1]圈奶牛Fencing the Cows /【模板】二维凸包 题目&#xff1a; 给定一些点&#xff0c;问围住所有点所用的围栏的长度 题解&#xff1a; 凸包模板题 凸包详细 代码&#xff1a; #include<iostream> #include<algorithm> #include<cstdio&g…

最小代价(区间dp)(ybtoj)

文章目录题目描述解析代码题目描述 解析 &#xff08;我觉得&#xff09;很难的dp 思路是真的没有想出来 关键在于dp的设计&#xff1a; dp[l][r]&#xff1a;[l,r]的最小价值 f[l][r][a][b]&#xff1a;把l到r之间除了数值在[a,b]之间的数全部消掉需要的最小价值 &#xff08…

P4783-[模板]矩阵求逆

正题 题目链接:https://www.luogu.com.cn/problem/P4783 题目大意 给出一个矩阵&#xff0c;求它的逆矩阵。 1≤n≤4001\leq n\leq 4001≤n≤400 解题思路 记给出矩阵PPP&#xff0c;记单位矩阵EEE。 PP−1E⇒P(EP−1)EP\times P^{-1}E\Rightarrow P\times (E\times P^{-1})…

.net core i上 K8S(六).netcore程序的service网络代理模式

正文上一章我们讲了pod的hostip模式&#xff0c;但在生产环境中&#xff0c;我们都是通过service来访问k8s集群的&#xff0c;service有两种模式来暴漏端口&#xff0c;今天我们来分享一下1.clusterIP模式我们在创建service的时候&#xff0c;默认创建的时clusterIP模式&#x…