C. Balance
time limit per test

3 seconds

memory limit per test

128 megabytes

input

standard input

output

standard output

Nick likes strings very much, he likes to rotate them, sort them, rearrange characters within a string... Once he wrote a random string of characters a, b, c on
a piece of paper and began to perform the following operations:

  • to take two adjacent characters and replace the second character with the first one,
  • to take two adjacent characters and replace the first character with the second one

To understand these actions better, let's take a look at a string «abc». All of the following strings can be obtained by performing one of the described operations
on «abc»: «bbc», «abb»,
«acc». Let's denote the frequency of a character for each of the characters a, band c as
the number of occurrences of this character in the string. For example, for string «abc»: |a|
= 1, |b| = 1, |c| = 1, and for string «bbc»:
|a| = 0, |b| = 2, |c|
= 1.

While performing the described operations, Nick sometimes got balanced strings. Let's say that a string is balanced, if the frequencies of each character differ by at most 1. That is  - 1 ≤ |a| - |b| ≤ 1,  - 1 ≤ |a| - |c| ≤ 1 и  - 1 ≤ |b| - |c| ≤ 1.

Would you help Nick find the number of different balanced strings that can be obtained by performing the operations described above, perhaps multiple times, on the given string s.
This number should be calculated modulo 51123987.

Input

The first line contains integer n (1 ≤ n ≤ 150) —
the length of the given string s. Next line contains the given string s.
The initial string can be balanced as well, in this case it should be counted too. The given string s consists only of characters a, b and c.

Output

Output the only number — the number of different balanced strings that can be obtained by performing the described operations, perhaps multiple times, on the given string s,
modulo 51123987.

Sample test(s)
input
4
abca
output
7
input
4
abbc
output
3
input
2
ab
output
1
Note

In the first sample it is possible to get 51 different strings through the described operations, but only 7 of
them are balanced: «abca», «bbca», «bcca»,
«bcaa», «abcc», «abbc»,
«aabc». In the second sample: «abbc», «aabc»,
«abcc». In the third sample there is only one balanced string — «ab»
itself.

题意:

给出一个最多150字符长的仅仅有a或b或c组成的字符串

对于每一个操作能够把前面一个字符变成后面一个字符或者把后面的一个字符变成前面一个字符 

即能够运行赋值语句 str[i+1] = str[i]; 或者 str[i] = str[i+1];

假设原字符串在运行若干次操作后变成一个a,b,c的字符数量相互不超过1的 字符串, 那么称得到的串为一个合法串

问合法串有多少个。

思路来源于:点击打开链接(推荐,写的好清晰)

思路:

依据性质。不可能交叉替换,如果输入的字符串是 A, 字符串 A'是 A串的同样字符压缩成一个后的结果 ,所以B是由A'一个字符扩展一个区间来的,状态 dp[i][na][nb][nc] 表示 以 A'中第i个字符为结尾的字符a数量为na, 字符b数量为nb, 字符c数量为nc的字符串的数量,防止一个B计算两次,规定下一个字符转移仅仅能用近期的来转移。

代码:

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <string>
#include <map>
#include <stack>
#include <vector>
#include <set>
#include <queue>
#define maxn 500005
#define MAXN 100005
#define mod 51123987
#define INF 0x3f3f3f3f
#define pi acos(-1.0)
#define eps 1e-12
typedef long long ll;
using namespace std; int n,m,ans,cnt,tot,flag;
int dp[155][55][55][55];
int next[155][3];
char s[200],s1[200]; int main()
{
int i,j,k,t;
while(~scanf("%d",&n))
{
scanf("%s",s+1);
tot=1;
s1[1]=s[1];
for(i=2; i<=n; i++)
{
if(s[i]!=s1[tot]) s1[++tot]=s[i];
}
memset(next,0,sizeof(next));
for(j=1; j<=tot; j++)
{
if(s1[j]=='a')
{
if(next[0][0]==0) next[0][0]=j;
}
else if(s1[j]=='b')
{
if(next[0][1]==0) next[0][1]=j;
}
else
{
if(next[0][2]==0) next[0][2]=j;
}
}
for(i=1; i<=tot; i++)
{
for(j=i; j<=tot; j++)
{
if(s1[j]=='a')
{
if(next[i][0]==0) next[i][0]=j;
}
else if(s1[j]=='b')
{
if(next[i][1]==0) next[i][1]=j;
}
else
{
if(next[i][2]==0) next[i][2]=j;
}
}
}
int na,nb,nc;
m=n/3+1;
dp[0][0][0][0]=1;
for(i=0; i<=tot; i++)
{
for(na=0; na<=m; na++)
{
for(nb=0; nb<=m; nb++)
{
for(nc=0; nc<=m; nc++)
{
if(next[i][0])
{
dp[next[i][0]][na+1][nb][nc]+=dp[i][na][nb][nc];
dp[next[i][0]][na+1][nb][nc]%=mod;
}
if(next[i][1])
{
dp[next[i][1]][na][nb+1][nc]+=dp[i][na][nb][nc];
dp[next[i][1]][na][nb+1][nc]%=mod;
}
if(next[i][2])
{
dp[next[i][2]][na][nb][nc+1]+=dp[i][na][nb][nc];
dp[next[i][2]][na][nb][nc+1]%=mod;
}
}
}
}
}
ans=0;
if(n%3==0)
{
for(i=1; i<=tot; i++)
{
ans+=dp[i][n/3][n/3][n/3];
ans%=mod;
}
}
else if(n%3==1)
{
for(i=1; i<=tot; i++)
{
ans+=dp[i][n/3+1][n/3][n/3];
ans+=dp[i][n/3][n/3+1][n/3];
ans+=dp[i][n/3][n/3][n/3+1];
ans%=mod;
}
}
else
{
for(i=1; i<=tot; i++)
{
ans+=dp[i][n/3+1][n/3+1][n/3];
ans+=dp[i][n/3+1][n/3][n/3+1];
ans+=dp[i][n/3][n/3+1][n/3+1];
ans%=mod;
}
}
printf("%d\n",ans);
}
return 0;
}
/*
4
abca
*/

Codeforces Beta Round #17 C. Balance (字符串计数 dp)的更多相关文章

  1. Codeforces Beta Round #17 C. Balance DP

    C. Balance 题目链接 http://codeforces.com/contest/17/problem/C 题面 Nick likes strings very much, he likes ...

  2. Codeforces Beta Round #17 D. Notepad (数论 + 广义欧拉定理降幂)

    Codeforces Beta Round #17 题目链接:点击我打开题目链接 大概题意: 给你 \(b\),\(n\),\(c\). 让你求:\((b)^{n-1}*(b-1)\%c\). \(2 ...

  3. Codeforces Beta Round #16 E. Fish (状压dp)(概率dp)

    Codeforces Beta Round #16 (Div. 2 Only) E. Fish 题目链接:## 点击打开链接 题意: 有 \(n\) 条鱼,每两条鱼相遇都会有其中一只吃掉对方,现在给你 ...

  4. Codeforces Beta Round #17 A - Noldbach problem 暴力

    A - Noldbach problem 题面链接 http://codeforces.com/contest/17/problem/A 题面 Nick is interested in prime ...

  5. Codeforces Beta Round #17 A.素数相关

    A. Noldbach problem Nick is interested in prime numbers. Once he read about Goldbach problem. It sta ...

  6. Codeforces Beta Round #17 D.Notepad 指数循环节

    D. Notepad time limit per test 2 seconds memory limit per test 64 megabytes input standard input out ...

  7. Codeforces Beta Round #51 D. Beautiful numbers 数位dp

    D. Beautiful numbers Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/55/p ...

  8. Codeforces Beta Round #13 C. Sequence (DP)

    题目大意 给一个数列,长度不超过 5000,每次可以将其中的一个数加 1 或者减 1,问,最少需要多少次操作,才能使得这个数列单调不降 数列中每个数为 -109-109 中的一个数 做法分析 先这样考 ...

  9. Codeforces Beta Round #72 (Div. 2 Only)

    Codeforces Beta Round #72 (Div. 2 Only) http://codeforces.com/contest/84 A #include<bits/stdc++.h ...

随机推荐

  1. 如何访问mvc 默认的错误页

    在ActionResult 中: public ActionResult Error() { return View("~/Views/Shared/Error.cshtml"); ...

  2. HRBUST 1200 装修

    $dp$. $dp[i]$记录铺满$3*i$的方案数.最后两列铺了$1$个$2*2$的和两个$1*1$的,或者是最后$1$列铺了$3$个$1*1$的. 因此$dp[i]=dp[i-1]+dp[i-2] ...

  3. struts2核心配置之Result

    result作用:在struts.xml中,使用<result>元素配置result逻辑视图和物理视图之间的映射 元素属性 属性 说明 是否必须 name 指定逻辑视图的名称(Action ...

  4. Web框架以及两种模式MVC,MTV

    一.Web框架的本质 众所周知,对于所有的Web应用,本质上其实就是一个socket服务端,用户的浏览器其实就是一个socket客户端. import socket def handle_reques ...

  5. 洛谷——P1958 上学路线_NOI导刊2009普及(6)

    P1958 上学路线_NOI导刊2009普及(6) 题目描述 你所在城市的街道好像一个棋盘,有a条南北方向的街道和b条东西方向的街道.南北方向的a条街道从西到东依次编号为l到a,而东西方向的b条街道从 ...

  6. ansible用playbook实现定期监控各机器磁盘和进程状态

    目标:用ansible定期监控各机器的磁盘空间状况 和进程运行状况 1)配置playbook脚本,实现对磁盘空间 和 特定进程运行状态的每日检查: 2)通过邮件插件,把检测结果发到ops邮箱: 一.p ...

  7. Redux 洋葱模型理解

    下面的代码会输出: A middleware1 开始C middleware2 开始E middleware3 开始======= G =======F middleware3 结束D middlew ...

  8. 【BZOJ 4567】【SCOI 2016】背单词

    http://www.lydsy.com/JudgeOnline/problem.php?id=4567 贪心. 任何不用第一种情况的方案吃的泡椒数都小于\(n^2\),所以最小泡椒数的方案一定不包含 ...

  9. 洛谷P2782 友好城市

    题目描述 有一条横贯东西的大河,河有笔直的南北两岸,岸上各有位置各不相同的N个城市.北岸的每个城市有且仅有一个友好城市在南岸,而且不同城市的友好城市不相同.没对友好城市都向政府申请在河上开辟一条直线航 ...

  10. JZYZOJ1527 [haoi2012]高速公路 线段树 期望

    http://172.20.6.3/Problem_Show.asp?id=1527 日常线段树的pushdown写挂,果然每次写都想得不全面,以后要注意啊……求期望部分也不熟练,和平均数搞混也是or ...