您當前位置>首頁 » 新聞資訊 » 技(jì)術(shù)分(fēn)享 >
第十一(yī)屆藍(lán)橋杯大(dà)賽軟件(jiàn)類省賽第二場(chǎng) H:←α♦ 子(zǐ)串分(fēn)值和(hé)
發表時(shí)間(jiān):2020-10-19
發布人(rén):葵宇科(kē)技(jì)
浏覽次數(shù):93
題解
列舉每個(gè)左端點(l), 每個(gè)字母自(××→λzì)力計(jì)算(suàn)供獻,列舉λ≥每個(gè)字母,在【l, n】 中找到(dào)該字母第一(yī)ε§次出現(xiàn)的(de)地(dì)位p, 則左端點為(wèi) l, 右短(duǎn)點 ∏在【p, n】的(de)子(zǐ)串都(dōu)獲得(de)該字母的(de)供獻,δ↔ε;供獻 n - p + 1。
例如(rú) :
- ccaba
- 當 l = 1 時(shí) :
- 列舉 ‘a’ , 第一(yī)次出現(xiàn)的(de)地(dì •₩)位 p = 3 供獻 5 - 3 + 1 = 3。
- 列舉 ‘b’ …
- 當 l = 2 …
序列自(zì)念頭保護下(xià)一(yī)字符地(dì)位, 複雜(zá)度 O(26 * ε$ n)
#pragma GCC optimize(2)
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<int, int> pii;
typedef vector<int> vi;
const int maxn = 2e5 + 10, mod = 1e9 + 7, inf = 0x3f3f3f3f;
char s[maxn]; int n, nxt[maxn][30];
void init()
{
for(int i = n; i >= 1; i--)
{
for(int j = 1; j <= 26; j++)
nxt[i-1][j] = nxt[i][j];
nxt[i-1][s[i]-'a'+1] = i;
}
}
int main()
{
ios::sync_with_stdio(0);
cin.tie(0);
cin >> s + 1;
n = strlen(s + 1);
init();
ll ans = 0;
for(int i = 1; i <= n; i++)
{
for(int j = 1; j <= 26; j++)
{
int p = nxt[i-1][j];
if(p == 0) continue;
ans += (n - p + 1);
}
}
cout << ans << endl;
return 0;
}