[AGC027E]ABBreviate
Description
AGC027E
给定一个仅由\(AB\)构成的字符串\(S\),给定两个操作,把\(AA\)换成\(B\),和把\(BB\)换成\(A\),问由这个字符串和任意次操作可以得到几种字符串。
Solution
AGC好难啊(摔
首先,如果不能操作,答案是\(0\)。
我们令\(A=1,B=2\),这样就会有一个美妙的性质:所有字符的和在操作的时候在模\(3\)意义下不变,我们记这个和为\(p(S)\)。然后我们可以得到一个结论:一个字符串\(S\)可以变成字符\(c\),当且仅当:\(S=c\)或\(p(S)=p(c)\)且\(S\)可以操作。这个条件的必要性很好证明,而充分性证明也比较简单:当\(S\)中的连续字符长度为\(2\)时,可以合并这两个字符,并且字符串长度减小;而当\(S\)中的连续字符长度大于\(2\)时,可以合并与不同字符相邻的那一端的字符(如果只有一种字符的话,自己证吧),字符串长度也减少;这样,字符串长度不断减少,一定可以达到\(1\),这时的\(S'=c\)。
这样问题就转化为:求出这样的\(T\)的数量:将\(S\)划分为\(|T|\)段,第\(i\)段的\(p\)值与\(T\)中的第\(i\)个字符的\(p\)值相等。我们可以贪心的构造这样的\(T\):让每一段的长度尽可能的小,如果分出\(|T|\)段后,剩余部分的\(p\)值为\(0\),那就是一个合法的\(T\)。
这样,就只需要一个简单的DP就行了我们递推的来做,\(f_i\)表示前\(i\)个字符看作一段的方案数,如果\(p(S_{i+1,n})=0\),那么\(i\)之后的位置可以看成是剩余的位置,\(f_i\)的初值为\(1\),否则为\(0\)。然后,我们可以在\(i\)之后加一段\(A\)或是\(B\),这样就要记录各个\(p\)值最后一次出现的位置,累加即可。具体实现看代码。
Code
#include <cstdio>
const int N = 1e5 + 10;
const int MOD = 1e9 + 7;
char s[N];
int a[N], n, f[N], nxt[3];
int main() {
scanf("%s", s);
for (; s[n]; ++n) a[n+1] = (a[n] + s[n] - 'a' + 1) % 3; // 求前缀和
// 判断是否可以操作
int flag = 0;
for (int i = 1; i < n; ++i) {
if (s[i] == s[i-1]) flag = 1;
}
if (!flag) {
puts("1");
return 0;
}
// 递推
f[n] = 1;
for (int i = 0; i < 3; ++i) nxt[i] = i == a[n] ? n : n + 1;
for (int i = n-1; i; --i) {
f[i] = a[i] == a[n]; // 后面可不可以不分段
for (int c = 1; c <= 2; ++c) {
(f[i] += f[nxt[(a[i]+c)%3]]) %= MOD; // 后面加一段A或B
}
nxt[a[i]] = i; // 记录该p值最后一次出现的位置
}
printf("%d\n", (f[nxt[1]] + f[nxt[2]]) % MOD);
return 0;
}
[AGC027E]ABBreviate的更多相关文章
- AGC027 E - ABBreviate
目录 题目链接 题解 代码 题目链接 AGC027 E - ABBreviate 题解 神仙啊 建议查看https://img.atcoder.jp/agc027/editorial.pdf 定义a ...
- java stopwatch 功能
C#中有一个stopwatch的功能,主要是用来监测程序执行时间的.java之前一直都在用如下方式完成: public static void main(String[] args) { long s ...
- gradle学习笔记
一直想着花时间学习下gradle,今天有空.入门一下.参考:极客学院gradle使用指南,官方文档:gradle-2.12/docs/userguide/installation.html,以及百度阅 ...
- Linux监控工具介绍系列——smem
smem工具介绍 smem是Linux系统上的一款可以生成多种内存耗用报告的命令行工具.与现有工具不一样的是smem可以报告实际使用的物理内存(PSS),这是一种更有意义的指标.可以衡量虚拟内存系统的 ...
- StringUtils工具类
StringUtils源码,使用的是commons-lang3-3.1包.下载地址 http://commons.apache.org/lang/download_lang.cgi 以下是String ...
- POJ2001Shortest Prefixes[Trie]
Shortest Prefixes Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 17683 Accepted: 768 ...
- StringUtils中的常用的方法
org.apache.commons.lang.StringUtils中常用的方法,这里主要列举String中没有,且比较有用的方法: 1. 检查字符串是否为空: static boolean isB ...
- Google C++ Style Guide
Background C++ is one of the main development languages used by many of Google's open-source project ...
- Apache Commons Lang
http://commons.apache.org/proper/commons-lang/javadocs/api-release/org/apache/commons/lang3/package- ...
随机推荐
- mysql 数据库优化的几种方法
1.选取最适用的字段属性 MySQL可以很好的支持大数据量的存取,但是一般说来,数据库中的表越小,在它上面执行的查询也就会越快.因此,在创建表的时候,为了获得更好的性能,我们可以将表中字段的宽度设得尽 ...
- ACM-ICPC 2018 徐州赛区网络预赛 Ryuji doesn't want to study
简单数学变换+线段树 简单数据结构签到题不解释 本来应该贴板子的,鉴于最近写代码太少了,而且由于要用两个线段树,平时板子都是一个的.以及板子在队友那.就当熟悉写代码,自己写了一下. #include ...
- spring-framework核心接口ApplicationContext
核心接口(ApplicationContext) 继承关系 继承接口: org.springframework.beans.factory.ListableBeanFactory:用于访问应用程序组件 ...
- VNote笔记本和画图
VNote笔记本 跨平台的,以markdown标记语言记录的文本文档.从sourceforget.org开源网址下载即可. 画图集成: 1.集成graphviz http://www.graphviz ...
- 题解【AcWing279】自然数拆分
题面 因为题目中说参与加法运算的数可以重复,由此可以想到完全背包计数问题. 完全背包计数问题与 \(01\) 背包计数问题只有一个不同: \(01\) 背包计数问题的第二维循环是倒叙循环,而完全背包计 ...
- 天兔修改登录页的title
1.将 /opt/lampp/htdocs/lepus/application/views/login.php 文件中 第6行 <title><?php echo $this-> ...
- Tensorflow中one_hot() 函数用法
官网默认定义如下: one_hot(indices, depth, on_value=None, off_value=None, axis=None, dtype=None, name=None) 该 ...
- 【HTML】iframe嵌套界面自适应,可高度自由收缩
最近在做网页时需要使iframe高度自适应,以提高用户体验,网上找了挺多都很复杂,最后找到了这个 HTML: <div class="main_page"> <i ...
- Mybatis- 基础知识
mybatis是一个java持久层框架,java中操作关系型 数据库用的是jdbc,mybatis是对jdbc的一个封装. 简介 iBATIS一词来源于"internet" ...
- python3练习100题——025
原题链接:http://www.runoob.com/python/python-exercise-example25.html 题目:求1+2!+3!+...+20!的和. 我的代码: s =[] ...