@AGC037 - E@ Reversing and Concatenating
@description@
给定一个长度为 N 且只包含小写字母的字符串 S ,你可以执行 k 次操作,每次操作你可以:
(1)将 S 翻转得到 T,将 S 与 T 拼接得到 U。
(2)从 U 中取出长度为 N 的子串 S',替换当前 S 进行下一轮迭代。
你需要求出 k 次操作后字典序最小的 S。
Constraints
1≤N≤5000, 1≤K≤10^9, |S|=N。保证 S 只包含小写字母。
Input
输入格式如下:
N K
S
Output
输出 K 次操作后字典序最小的字符串。
Sample Input 1
5 1
bacba
Sample Output 1
aabca
S=bacba, T=abcab, U=bacbaabcab, S′=aabca。
Sample Input 2
10 2
bbaabbbaab
Sample Output 2
aaaabbaabb
@solution@
是我太傻。。。
k 这么大,是否一定操作过后字符串就恒定不变呢?
那么我们不妨大胆猜测,一定操作过后,字符串变成只含有最小字母 mn 的字符串("aaa...a" 型)。
最少多少次呢?假如我们 S 有一个以 mn 结尾的长度为 p 的连续段,则 U 的中间就有 2*p 的连续段。
假如我们选择以这 2*p 个 mn 为结尾的 S',我们又可以得到 4*p 个 mn 结尾的 S'' 等。即:mn 的数量可以呈指数级增长。
同时,只有末尾的连续段才会有这个性质,不在末尾的甚至不会与另一个字符串 T 产生关系。
当 k > log|S| 时,我们可以直接输出答案了。剩下的只需要讨论 k <= log|S| 的时候
根据字典序的定义,我们想让前面的字符尽可能地字典序小。
而最后一次我们可以让 mn 直接放在字符串的开头,这样 mn 越多显然越优秀。
于是我们的贪心策略就是:枚举第一次操作选择的字符串,然后按照上面所说,将末尾的连续段翻倍再翻倍。
最后一次再把连续段作为开头截取下来,得到最终的字符串。
你以为要暴力模拟过程?NONONO!
我们在最后一次之前,末尾连续段之前的那些 “冗余” 部分的相对顺序是不会变的,保持着在原串中的顺序。
而最后一次只是把那些 “冗余” 部分翻转一下,所以可以直接得到最终字符串。
因为更新答案是暴力比较的,所以时间复杂度是 O(n^2) 的。
@accepted code@
#include<cstdio>
#include<algorithm>
using namespace std;
const int MAXN = 5000;
char str[2*MAXN + 5], mn;
char ans[MAXN + 5];
int N, K;
bool check(char *S) {
for(int i=1;i<=N;i++)
if( ans[i] != S[i] )
return ans[i] > S[i];
return false;
}
void update(char *S) {
for(int i=1;i<=N;i++)
ans[i] = S[i];
}
int pw[35];
char S[MAXN + 5];
int main() {
pw[0] = 0, pw[1] = 2;
for(int i=2;i<=30;i++) pw[i] = 2*pw[i-1];
scanf("%d%d%s", &N, &K, str + 1);
mn = 'z';
for(int i=1;i<=N;i++)
mn = min(mn, str[i]), ans[i] = 'z';
if( K >= 30 ) {
for(int i=1;i<=N;i++)
putchar(mn);
}
else {
for(int i=1;i<=N;i++)
str[2*N - i + 1] = str[i];
int cnt = 0;
for(int i=1;i<N;i++)
cnt = (str[i] == str[i-1]) ? cnt + 1 : 1;
for(int i=N;i<=2*N;i++) {
cnt = (str[i] == str[i-1]) ? cnt + 1 : 1;
if( pw[K - 1] <= N/cnt ) {
for(int j=1;j<=pw[K-1]*cnt;j++)
S[j] = str[i];
int k = i - cnt;
if( K == 1 ) k = i;
for(int j=pw[K-1]*cnt+1;j<=N;j++)
S[j] = str[k--];
}
else {
for(int j=1;j<=N;j++)
S[j] = str[i];
}
if( check(S) ) update(S);
}
puts(ans + 1);
}
}
@details@
可能是自己还不大擅长贪心。。。
贪心太难了。。。
@AGC037 - E@ Reversing and Concatenating的更多相关文章
- [Atcoder AGC037E]Reversing and Concatenating
题目大意:有一个长度为$n$的字符串$S$,有$k$次操作,每次操作为把$S$变为$SS^R$(即翻转后再接在一起),然后从中选取一段长度为$n$的字串.问$k$次操作后,字典序最小的一种是什么.$n ...
- AtCoder Grand Contest 037
Preface 这篇咕了可能快一个月了吧,正好今天晚上不想做题就来补博客 现在还不去复习初赛我感觉我还是挺刚的(微笑) A - Dividing a String 考虑最好情况把每个字符串当作一个来看 ...
- AtCoder Grand Contest 037 简要题解
从这里开始 题目目录 Problem A Dividing a String 猜想每段长度不超过2.然后dp即可. 考虑最后一个长度大于等于3的一段,如果划成$1 + 2$会和后面相同,那么划成$2 ...
- Assembler : The Basics In Reversing
Assembler : The Basics In Reversing Indeed: the basics!! This is all far from complete but covers ab ...
- 02-线性结构2 Reversing Linked List
由于最近学的是线性结构,且因数组需开辟的空间太大.因此这里用的是纯链表实现的这个链表翻转. Given a constant K and a singly linked list L, you are ...
- 单链表反转的递归实现(Reversing a Linked List in Java, recursively)
转自Reversing a Linked List in Java, recursively There's code in one reply that spells it out, but you ...
- PAT1074 Reversing Linked List (25)详细题解
02-1. Reversing Linked List (25) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 Standard 作者 CHEN, Yue ...
- [Reversing.kr] Easy Crack Writeup
题目:http://reversing.kr/ Easy Crack IDA打开.分析可知Sub_401080是关键函数.F5后. 当满足 则跳转成功.拼接后得到flag flag: Ea5yR3ve ...
- PAT 1074 Reversing Linked List[链表][一般]
1074 Reversing Linked List (25)(25 分) Given a constant K and a singly linked list L, you are suppose ...
随机推荐
- H5C3--边框阴影box-shadow
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- H5C3--视频播放器
CSS css.css body { ; ; font-family: 'microsoft yahei', 'Helvetica', simhei, simsun, sans-serif; back ...
- JavaScript中this的指向2(转载)
1. 每个函数都包含两个非继承而来的方法:call()方法和apply()方法. 2. 相同点:这两个方法的作用是一样的. 都是在特定的作用域中调用函数,等于设置函数体内this对象的值,以扩充函数赖 ...
- bootstrap-fileinput详细说明与使用
介绍 bootstrap-fileinput是一款非常优秀的HTML5文件上传插件,支持文件预览.多文件上传等一系列特性. 一款非常优秀的HTML5文件上传插件,支持bootstrap 3.x 和4. ...
- Python执行时间的计算方法小结
Python执行时间的计算方法小结 首先说一下我遇到的坑,生产上遇到的问题,我调度Python脚本执行并监控这个进程,python脚本运行时间远远大于python脚本中自己统计的程序执行时间. 监控p ...
- GIT → 00:GIT学习大纲
1. 学习版本控制的原因 1.1 没有版本控制出现的问题 1.2 版本控制的简介 1.3 版本控制工具 2. Git 和 Svn 比较 2.1 SVN介绍 2.1.1 SVN简介 2.1.2 SVN基 ...
- 2019.8.7 NOIP模拟测试14 反思总结
先扔代码 调完自闭网络流了,新一轮考试前看看能不能赶完…… 考得极其爆炸,心态也极其爆炸,真的是认识到自己能力上的不足 思维跑偏,代码能力差 就这样吧,再努力努力,不行就AFO T1旋转子段: 因为我 ...
- linux应用系统日志
在一个典型的LAMP(Linux+Apache+Mysql+Perl)应用环境里: Apache & Nginx; 查找访问和错误日志, 直接找 5xx 错误, 再看看是否有 limit_zo ...
- Leetcode91.Decode Ways解码方法
一条包含字母 A-Z 的消息通过以下方式进行了编码: 'A' -> 1 'B' -> 2 ... 'Z' -> 26 给定一个只包含数字的非空字符串,请计算解码方法的总数. 示例 1 ...
- 上传同步github
…or create a new repository on the command line echo "# testproject" >> README.md ...