【CF594E】Cutting the Line 【贪心】【Lyndon Word】【扩展kmp】

传送门

题意:给一个字符串SSS和正整数kkk,将SSS分成最多kkk段,每段不变或翻转,使得最后的字典序最小。

∣S∣≤5×106|S|\leq5\times10^6S5×106

发现不翻转可以看成拆成若干单字符分别翻转,所以先分析一下必须翻转的情况

把原串翻转记为SRS^RSR,然后我们要求的是不断剪掉SRS^RSR的后缀然后依次拼起来

这样最终串的第一段是SRS^RSR的一个后缀,所以最终串的开头一定有SRS^RSR的最小后缀,但不一定是最小后缀作为第一段,因为最小后缀可能会在前面作为非后缀出现

显然这个“最小后缀”是Lyndon分解后的最后一段,记为sss 我们希望开头的sss尽量多

那么SRS^RSR可表示为a1s+t1+a2s+t2+...+ans+tn+asa_1s+t_1+a_2s+t_2+...+a_ns+t_n+asa1s+t1+a2s+t2+...+ans+tn+as(和Lyndon分解没有关系)

首先可以一刀把asasas砍掉,然后找到a1∼ana_1\sim a_na1an中最大的砍下来 发现这第二段是砍掉asasas后的最小后缀,相当于是下一轮的第一段

整理一下,对SRS^RSR进行Lyndon分解并合并相等段,这个Duval的时候魔改一下就可以了

然后依次砍掉最后一段并让k−1k-1k1

注意我们假设了必须翻转,如果我们发现有连续一段的长度为111的串,相当于这一段不翻转,只需要一步

这个流程需要砍掉两段(只是后面一段和下一步的第一段重合了),所以需要k>2k>2k>2

完了之后有k≤2k \leq 2k2,如果剩下的只有一段直接大力讨论掉

如果k=1k=1k=1,SSSSRS^RSR取个min⁡\minmin即可

如果k=2k=2k=2,相当于分两段大力讨论 注意是针对原串

  1. 前面后面都不翻 就是原串
  2. 只翻后面

我们考虑找到最优的位置

从左到右循环,设当前最优位置为cutcutcut,需要更新的位置为iii 注意cut<icut<icut<i

在这里插入图片描述
(橙色部分为反串,TTTSRS^RSR)

我们希望比较两个串的大小 所以从cutcutcut开始找到第一个不同的位置比较大小

首先求出Scut∼i−1S_{cut\sim i-1}Scuti1TTT的最长公共前缀,可以先跑一个exKMP,求出SSScutcutcut开始的后缀与TTT的最长公共前缀后和i−cuti-cuticutmin⁡\minmin

如果把蓝色部分顶满了,再加上后面的部分

TTTi−cuti-cuticut开始的后缀与TTT的最长公共前缀与n−i+1n-i+1ni+1min⁡\minmin

然后讨论一下找到第一个不同的字符比较大小即可

  1. 翻前面,后面不管

继续从SRS^RSR的结尾截后缀,设截取的后缀为TTT

考虑分解后的最后一个Lyndon串sss,TTT一定以sss开头,也以sss结尾

根据意识流,TTT一定不会只取一个分解后的LW的一部分,也不会把两个相等的LW隔开

TTT开始的第一段为s′s's,所以ssss′s's的前缀

然后有若干个s′s's接在后面,这些s′s's后的第一个设为ttt

根据Lyndon分解的定义,t≤s′t \leq s'ts。而如果t<s′t <s't<s,那么从ttt开始截取后缀会比TTT小,与定义矛盾

所以TTT一定是s′+s′+...+s′+s+s+...+ss'+s'+...+s'+s+s+...+ss+s+...+s+s+s+...+s的形式

把上面剩下的 Lyndon分解合并相等段 的倒数第二段提出来,如果sss是它的前缀,说明倒数第二段是s′s's,此时分类讨论翻后面两段或者只翻最后一段;如果不是说明s′s's不存在,只能翻最后一段

第二段和反串取min⁡\minmin接在后面

复杂度O(n)O(n)O(n)

如果用std::string的话,要注意A=A+BA+=B复杂度不同……

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cctype>
#include <string>
#include <algorithm>
#define MAXN 10000005
using namespace std;
string s,t,ts,ans;
int pos[MAXN],len[MAXN],tot;
inline string reverse(string s)
{string t;t.resize(s.size());int n=s.size();for (int i=0;i<n;i++) t[n-i-1]=s[i];return t;
}
void Duval(const string& s)
{int n=s.size();for (int i=0;i<n;){int j=i,k=i+1;while (s[j]<=s[k]) {if (s[j]==s[k]) ++j;else j=i; ++k;}len[++tot]=k-j;while (i<=j){pos[tot]=i+k-j-1;i+=k-j;}}
}
int p[MAXN];
void Exkmp(const string& s)
{int n=s.size();int mid=0,mx=0;p[0]=n;for (int i=1;i<n;i++){if (i<=mx) p[i]=min(p[i-mid],mx-i+1);while (s[i+p[i]]==s[p[i]]) ++p[i];if (i+p[i]-1>mx) mid=i,mx=i+p[i]-1;}
}
int main()
{ios::sync_with_stdio(false);cin>>s;t=reverse(s);int k;cin>>k;if (k==1) return cout<<min(s,t),0;Duval(t);pos[0]=-1;while (k>2&&tot){if (len[tot]==1){while (tot&&len[tot]==1) ans+=t.substr(pos[tot-1]+1,pos[tot]-pos[tot-1]),--tot;--k;}else ans+=t.substr(pos[tot-1]+1,pos[tot]-pos[tot-1]),--k,--tot;}if (tot==0) return cout<<ans,0;if (tot==1){string tmp=t.substr(0,pos[1]+1);tmp=min(tmp,reverse(tmp));cout<<ans+tmp;return 0;}s=reverse(t=t.substr(0,pos[tot]+1));ts=t+"#"+s;Exkmp(ts);string tmp=min(s,t);int cut=0,n=s.size();for (int i=1;i<n;i++){int cl=min(i-cut,p[n+1+cut]);if (cut+cl==i) cl+=p[cl];cl=min(cl,n-cut);if ((cl<i-cut? s[cut+cl]:t[cut+cl-i])<t[cl]) cut=i;}tmp=min(tmp,s.substr(0,cut)+t.substr(0,n-cut));string las=t.substr(pos[tot-1]+1,len[tot]);string lass=t.substr(pos[tot-2]+1,len[tot]);int st=pos[tot-1]+1;string tt=t.substr(st,n-st);string res=t.substr(0,n-tt.size());tt+=min(res,reverse(res));tmp=min(tmp,tt);if (las==lass){st=pos[tot-2]+1;tt=t.substr(st,n-st);res=t.substr(0,n-tt.size());tt=tt+min(res,reverse(res));tmp=min(tmp,tt);}cout<<ans+tmp;return 0;
}

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

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

相关文章

一份好的工作总结才能帮你升职加薪

这里是Z哥的个人公众号每周五早8点 按时送达当然了&#xff0c;也会时不时加个餐&#xff5e;我的第「79」篇原创敬上最近有点忙&#xff0c;搬出之前攒的一篇文章来应急一下。一篇能助你挣更多钱的文章。好了&#xff0c;下面开始。我的读者们大部分是互联网行业的&#xff0c…

腾讯开源软件镜像站上线

腾讯开源软件镜像站(Tencent Open Source Mirror Site)已于近日上线&#xff0c;其官方名称为「腾讯云软件源」&#xff0c;由腾讯云提供支持。官方表示搭建此开源镜像站的目的在于宣传自由软件的价值&#xff0c;提高自由软件社区文化氛围&#xff0c;推广自由软件在国内的应用…

ASP.NET Core on K8S学习初探(2)

“ [LOG] ASP.NET Core on K8S Starting...”在上一篇《单节点环境搭建》中&#xff0c;通过Docker for Windows在Windows开发机中搭建了一个单节点的K8S环境&#xff0c;接下来就是动人心弦的部署ASP.NET Core API到K8S了。但是&#xff0c;在部署之前&#xff0c;我还是把基本…

Educational Codeforces Round 96 E. String Reversa 线段树模拟序列交换

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 思路&#xff1a; 与上一篇题解大同小异&#xff0c;无非就是不需要枚举排列了。 // Problem: E. String Reversal // Contest: Codeforces - Educational Codeforces Round 96 (Rated for Div. 2) // URL:…

ASP.NET Core on K8S学习初探(1)

“ [LOG] ASP.NET Core on K8S Starting...”01—写在之前当近期的一个App上线后&#xff0c;发现目前的docker实例&#xff08;应用服务BFF中台服务工具服务&#xff09;已经越来越多了&#xff0c;而我司目前没有专业的运维人员&#xff0c;发现运维的成本逐渐开始上来&#…

.NET Core IdentityServer4实战 第Ⅳ章-集成密码登陆模式

回顾下ClientCredentials模式&#xff0c;在ReSourceApi中定义了我们公开服务&#xff0c;第三方网站想要去访问ReSourceApi则需要在身份验证服务中获取toekn&#xff0c;根据token的内容&#xff0c;硬编码去访问公开服务&#xff08;ResApi&#xff09;,这个还是非常简单的&a…

Codeforces Round #656 (Div. 3) F. Removing Leaves 贪心 + 模拟

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 思路&#xff1a; 首先有一个贪心策略就是每次都找一个叶子节点最多的点&#xff0c;让后删掉他的kkk个叶子节点&#xff0c;现在我们就来考虑如何模拟这个过程。 我们整一个vector<set<int>>ve…

.NetCore中三种注入生命周期的思考

.NetCore彻底诠释了“万物皆可注入”这句话的含义&#xff0c;在.NetCore中到处可见注入的使用。因此core中也提供了三种注入方式的生命周期使用&#xff0c;分别是&#xff1a;AddTransient&#xff1a;每次请求&#xff0c;都获取一个新的实例。即使同一个请求获取多次也会是…

兰州大学第一届 飞马杯 体育课排队 二分 + 最大流 + 输出路径

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 思路&#xff1a; 读懂题就会辣&#xff0c;经典模型了&#xff0c;二分时间&#xff0c;让后将其转换成二分图&#xff0c;左边是人&#xff0c;右边是位置&#xff0c;能在规定时间到的连边&#xff0c;跑…

从零开始实现ASP.NET Core MVC的插件式开发(一) - 使用Application Part动态加载控制器和视图...

如果你使用过一些开源CMS的话&#xff0c;肯定会用过其中的的插件化功能&#xff0c;用户可以通过启用或者上传插件包的方式动态添加一些功能&#xff0c;那么在ASP.NET Core MVC中如何实现插件化开发呢&#xff0c;下面我们来探究一下。本系列只是笔者的一些尝试&#xff0c;并…

兰州大学第一届 飞马杯 ★★快乐苹果树★★ 树链剖分 + 懒标记 + 树状数组

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 思路&#xff1a; 第一次听说树链剖分能在fa[top[i]]fa[top[i]]fa[top[i]]的地方加懒标记&#xff0c;学到了学到了。 首先不能被题目吓住&#xff0c;这个题目仔细剖析一下不难发现一些性质&#xff1a; 以…

.NET分布式框架 | Orleans 知多少

引言公司物联网项目集成Orleans以支持高并发的分布式业务&#xff0c;对于Orleans也是第一次接触&#xff0c;本文就分享下个人对Orleans的理解。这里先抛出自己的观点&#xff1a;Orleans 是一个支持有状态云生应用/服务水平伸缩的基于Virtual Actor 模型的.NET分布式框架。下…

nowcoder 牛牛的最大兴趣组 质因子 + 思维

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 思路&#xff1a; 首先nnn很小的话可以暴力连边&#xff0c;让后染个色求一个颜色最多的即可。但是这个题显然不行&#xff0c;由于是三次方&#xff0c;所以考虑质因子入手。 首先很容易就能想到将所有的数…

ASP.NET Core on K8S学习初探(3)部署API到K8S

“ 终于可以部署ASP.NET Core到K8S中了...”在上一篇《基本概念快速一览》中&#xff0c;我们把基本的一些概念快速地简单地不求甚解地过了一下&#xff0c;本篇开始我们会将ASP.NET Core WebAPI部署到K8S&#xff0c;从而结束初探的旅程。01—准备一个WebAPI这里准备一个空的A…

K-D Tree学习笔记

引入 K-D Tree 是一种处理高维空间的数据结构。 支持O(nk−1k)O(n^{\frac {k-1}k})O(nkk−1​)查询给定超矩形内的点的信息&#xff0c; kkk 为维数。 可以用替罪羊树的思想实现动态插入。 但其实用的最多的是在错误的复杂度内查询奇奇怪怪的点信息。 构建 K-D Tree 是平…

Codeforces Round #633 B. Edge Weight Assignment 结论题 + dp

传送门 文章目录题意&#xff1a;思路&#xff1a;题意&#xff1a; 思路&#xff1a; 首先考虑最小值&#xff0c;如果从一个叶子结点出发到任意叶子的距离都为偶数&#xff0c;那么只需要一个值就可以满足条件。如果有奇数的&#xff0c;考虑111 ^ 222 ^ 303030&#xff0c;…

ASP.NET Core Web API中使用Swagger

本节导航Swagger介绍在ASP.NET CORE 中的使用swagger在软件开发中,管理和测试API是一件重要而富有挑战性的工作。在我之前的文章《研发团队,请管好你的API文档》 也专门阐述了通过文档管理工具,来保证API文档和代码的一致性,这样更加有助于团队的协作。以往我们总是通过第三方平…

【ZJOI2015】幻想乡战略游戏【点分树】【带权重心】

题意&#xff1a;nnn个点带边权的树&#xff0c;动态修改点权viv_ivi​&#xff0c;最小化 钦定一个点xxx 后 ∑idist(x,i)∗vi\sum\limits_{i} dist(x,i)*v_ii∑​dist(x,i)∗vi​的值。 n,q≤105n,q \leq10^5n,q≤105&#xff0c;度数不超过202020 限制度数的树上的一些诡异…

NuGet是什么?理解与使用

如果你了解python&#xff0c;那么它类似pip。如果你了解nodejs&#xff0c;那么它类似npm。如果你了解ruby&#xff0c;那么它类似gem。对&#xff0c;它就是一个包&#xff08;package&#xff09;管理平台&#xff0c;确切的说是 .net平台的包管理工具&#xff0c;它提供了一…

P2468 [SDOI2010]粟粟的书架 主席树 + 二分 + 二维前缀和

传送门 文章目录题意&#xff1a;题意&#xff1a;题意&#xff1a; 给你一个n∗mn*mn∗m的矩阵&#xff0c;每次询问一个矩形&#xff0c;左上角是(x1,y1)(x_1,y_1)(x1​,y1​)&#xff0c;右上角是(x2,y2)(x_2,y_2)(x2​,y2​)&#xff0c;和一个hhh&#xff0c;问是否能选出…