把串 s s s分为 x x x段,设每一段的 D : K = L \rm D:K=L D:K=L,那么串 s s s的 D : K \rm D:K D:K必然也为 L L L
所以对于每个前缀,只需要检验一下在比率 L \rm L L下至多能分成几段即可
进一步的,如果 s [ 1.. i ] s[1..i] s[1..i]的比率为 L L L, s [ 1... j ] s[1...j] s[1...j]的比率为 L L L,那么 s [ i + 1... j ] s[i+1...j] s[i+1...j]的比率也必然为 L L L
所以可以贪心,一旦比率为 L L L就马上截断
暴力判断的话单次复杂度 O ( n ) O(n) O(n),显然无法通过
没办法,发现最多只有 n n n个本质不同的比例,用 m a p map map存一下
然后对每个比例暴力做一遍更新答案,这样就是调和级数的复杂度.
#include <bits/stdc++.h>
using namespace std;
const int maxn = 5e5+10;
int t,n,d[maxn],k[maxn],res[maxn];
char a[maxn];
typedef pair<int,int>p;
map<p,int>mp;
int gcd(int a,int b){ return b==0?a:gcd(b,a%b); }
int main()
{
cin >> t;
while( t-- )
{
mp.clear();
scanf("%d%s",&n,a+1 );
for(int i=1;i<=n;i++)
{
d[i] = d[i-1], k[i] = k[i-1]; res[i] = 1;
if( a[i]=='D' ) d[i]++;
if( a[i]=='K' ) k[i]++;
int yin = gcd( d[i],k[i] );
if( mp.count( p(d[i]/yin,k[i]/yin) ) ) continue;
mp[ p(d[i]/yin,k[i]/yin) ] = 1;
}
for(auto vv:mp )
{
p v = vv.first;
int D = v.first, K = v.second, ans = 0, now = 1;
while( true )
{
int flag = 0;
for(int i=1,j=now+D+K-1;j<=n;i++,j=j+D+K)
{
int td = d[j]-d[now-1], tk = k[j]-k[now-1];
int yin = gcd( td,tk );
if( td/yin!=D || tk/yin!=K ) continue;
flag = 1, ans++, now = j+1;
res[now-1] = max( res[now-1],ans );
}
if( !flag ) break;
}
}
for(int i=1;i<=n;i++) printf("%d%c",res[i],i==n?'\n':' ');
}
}