牛客 A.托米的字符串(求期望、前缀)

题目链接:点击这里
牛客 A.托米的字符串(求期望、前缀)_第1张图片

设随机变量 X X X为 “元音 ( a , e , i , o , u , y ) (a,e,i,o,u,y) (a,e,i,o,u,y)字母占子串长度比”,求 X X X 的期望。

长度为 1 1 1 的子串有 n n n 个,

长度为 2 2 2 的子串有 n − 1 n-1 n1 个,

长度为 3 3 3 的子串有 n − 2 n-2 n2 个,

… …

长度为 n n n 的子串有 1 1 1 个,

所以,子串总共有 n ( n + 1 ) 2 \frac{n(n+1)}{2} 2n(n+1)个。

随机取一个子串,等概率,即 P ( X ) = 1 n ( n + 1 ) 2 P(X) = \frac{1}{\frac{n(n+1)}{2}} P(X)=2n(n+1)1

l e g i l i m e n s legilimens legilimens 为例进行推导, X X X 的分布列为:

长度为 1 1 1 的子串:

X 0 1 \frac{0}{1} 10 1 1 \frac{1}{1} 11 0 1 \frac{0}{1} 10 1 1 \frac{1}{1} 11 0 1 \frac{0}{1} 10 1 1 \frac{1}{1} 11 0 1 \frac{0}{1} 10 1 1 \frac{1}{1} 11 0 1 \frac{0}{1} 10 0 1 \frac{0}{1} 10
P 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551

长度为 2 2 2 的子串:

X 1 2 \frac{1}{2} 21 1 2 \frac{1}{2} 21 1 2 \frac{1}{2} 21 1 2 \frac{1}{2} 21 1 2 \frac{1}{2} 21 1 2 \frac{1}{2} 21 1 2 \frac{1}{2} 21 1 2 \frac{1}{2} 21 0 2 \frac{0}{2} 20
P 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551 1 55 \frac{1}{55} 551

… …

长度为 10 10 10 的子串:

X 4 10 \frac{4}{10} 104
P 1 55 \frac{1}{55} 551

E ( X ) = ∑ X i P i = 1 55 ( 4 1 + 8 2 + . . . . . . + 4 10 ) E(X) = \sum X_iP_i = \frac{1}{55}(\frac{4}{1}+\frac{8}{2}+......+\frac{4}{10}) E(X)=XiPi=551(14+28+......+104) 可得到一般结论如下:

s u m [ i ] sum[ i ] sum[i] 表示前 i i i 个字母中元音字母个数,

令长度为 i i i 的子串中元音字母出现的个数之和为 f [ i ] f[ i ] f[i]

f [ 1 ] = s u m [ n ] f[ 1 ]=sum[ n ] f[1]=sum[n]

f [ 2 ] = f [ 1 ] + s u m [ n − 1 ] − s u m [ 1 ] f[ 2 ]=f[ 1 ]+sum[n−1]−sum[ 1 ] f[2]=f[1]+sum[n1]sum[1](即在 f [ 1 ] f[1] f[1] 的基础上, s u m [ n − 1 ] − s u m [ 1 ] sum[n−1]−sum[1] sum[n1]sum[1] 这部分要再算一遍)

f [ 3 ] = f [ 2 ] + s u m [ n − 2 ] − s u m [ 2 ] f[ 3 ]=f[ 2 ]+sum[n−2]−sum[ 2 ] f[3]=f[2]+sum[n2]sum[2](即在 f [ 2 ] f[2] f[2] 的基础上, s u m [ n − 2 ] − s u m [ 2 ] sum[n−2]−sum[2] sum[n2]sum[2] 这部分要再算一遍)

⋯⋯

最后答案是 ∑ ( f [   i   ] / i ) n ( n + 1 ) / 2 \frac{\sum \left( f[\ i\ ]\right/i)}{n(n+1)/2} n(n+1)/2(f[ i ]/i)

AC代码如下:

#include
using namespace std;
long long sum[1000010], f[1000010]; 

int main()
{
    string s;
    cin>>s;
    int len = s.length();
    
    for(int i = 0, j = 1; i < len; j++, i++)
    {
    	if(s[i]=='a'||s[i]=='e'||s[i]=='i'||s[i]=='o'||s[i]=='u'||s[i]=='y')
    		sum[j] = sum[j-1] + 1;
    	else
    		sum[j] = sum[j-1];
	}
	
	for(int i = 1; i <= len; i++)
		f[i] = f[i-1] + sum[len-i+1] - sum[i-1];
	
	double ans = 0;
	for(int i = 1; i <= len; i++)
		ans += f[i]*1.0/i;
	
	printf("%0.9f\n", ans/(1.0*len*(len+1)/2.0));
	return 0;
}

你可能感兴趣的:(数学思维,概率与数学期望)