热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

CFEDU112D–SayNotoPalindromes的解题方法

本文介绍了解决CFEDU112D-SayNotoPalindromes问题的方法。通过枚举abc的六种排列,并利用前缀和求解最小改动次数,可以得到满足要求的字符串。具体方法包括记录前i个字母中(a,b,c)在(0,1,2)三个位置的个数,并计算出指定区间内满足要求的字符串的个数。

D - Say No to Palindromes


枚举

可观察到只有类似 abcabcabcabc..., bacbacbac... 等 abc 三个字母都循环出现才满足要求

可记录 \(cnt[i][j][k]\),前 \(i\) 个中 \(a, b, c\) 分别在 模 \(3\)\(0,1,2\) 的个数

因此枚举 abc 的 6 种排列,用前缀和求最小改动次数即可

#include
#include
#include
#include
#include
using namespace std;
typedef long long ll;
const int N = 2e5 + 10;
string s;
int cnt[N][3][3];//cnt[i][j][k]为前i个字母中(a, b, c)分别在(0, 1, 2)三个位置的个数
int n, m;
int l, r;
int solve(int a, int b, int c)
{
int cnt_a = cnt[r][a][0] - cnt[l-1][a][0];
int cnt_b = cnt[r][b][1] - cnt[l-1][b][1];
int cnt_c = cnt[r][c][2] - cnt[l-1][c][2];
return r - l + 1 - cnt_a - cnt_b - cnt_c;
}
int main()
{
ios::sync_with_stdio(false), cin.tie(0), cout.tie(0);
cin >> n >> m;
cin >> s;
s = " " + s;
for (int i = 1; i {
for (int j = 0; j <3; j++)
for (int k = 0; k <3; k++)
cnt[i][j][k] = cnt[i-1][j][k];
char ch = s[i];
int t = i % 3;
if (ch == 'a')
cnt[i][0][t]++;
else if (ch == 'b')
cnt[i][1][t]++;
else
cnt[i][2][t]++;
}

while(m--)
{
cin >> l >> r;
int ans = 1e9;
ans = min(ans, solve(0, 1, 2));
ans = min(ans, solve(0, 2, 1));
ans = min(ans, solve(1, 0, 2));
ans = min(ans, solve(1, 2, 0));
ans = min(ans, solve(2, 0, 1));
ans = min(ans, solve(2, 1, 0));
cout < }
return 0;
}


推荐阅读
author-avatar
你们聊交给我
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有