1. 程式人生 > >BZOJ2134: 單選錯位(期望亂搞)

BZOJ2134: 單選錯位(期望亂搞)

www. ring eof www span problems () led pascal

Time Limit: 10 Sec Memory Limit: 259 MB
Submit: 1101 Solved: 851
[Submit][Status][Discuss]

Description

技術分享圖片

Input

n很大,為了避免讀入耗時太多, 輸入文件只有5個整數參數n, A, B, C, a1, 由上交的程序產生數列a。 下面給出pascal/C/C++的讀入語句和產生序列的語句(默認從標準輸入讀入): // for pascal readln(n,A,B,C,q[1]); for i:=2 to n do q[i] := (int64(q[i-1]) * A + B) mod 100000001; for i:=1 to n do q[i] := q[i] mod C + 1; // for C/C++ scanf("%d%d%d%d%d",&n,&A,&B,&C,a+1); for (int i=2;i<=n;i++) a[i] = ((long long)a[i-1] * A + B) % 100000001; for (int i=1;i<=n;i++) a[i] = a[i] % C + 1; 選手可以通過以上的程序語句得到n和數列a(a的元素類型是32位整數), n和a的含義見題目描述。 2≤n≤10000000, 0≤A,B,C,a1≤100000000

Output

輸出一個實數,表示gx期望做對的題目個數,保留三位小數。

Sample Input

3 2 0 4 1

Sample Output

1.167
【樣例說明】
a[] = {2,3,1}
正確答案 gx的答案 做對題目 出現概率
{1,1,1} {1,1,1} 3 1/6
{1,2,1} {1,1,2} 1 1/6
{1,3,1} {1,1,3} 1 1/6
{2,1,1} {1,2,1} 1 1/6
{2,2,1} {1,2,2} 1 1/6
{2,3,1} {1,2,3} 0 1/6
共有6種情況,每種情況出現的概率是1/6,gx期望做對(3+1+1+1+1+0)/6 = 7/6題。(相比之下,lc隨機就能期望做對11/6題)

HINT

Source

時隔19260817年之後第一次自己做出BZOJ的題目QWQ

我的思路:

首先對於每一個位置的期望都是獨立的

然後就可以推出這個位置和它前一個位置的貢獻

為$min(1/a[i],1/a[pre(i)])$

其實挺顯然的

#include<cstdio>
#include<cstring>
#include<algorithm>
//#define getchar() (p1==p2&&(p2=(p1=buf)+fread(buf,1,MAXN,stdin),p1==p2)?EOF:*p1++)
using namespace std; const int MAXN=1e7+10; char buf[1<<20],*p1=buf,*p2=buf; inline int read() { char c=getchar();int x=0,f=1; while(c<0||c>9){if(c==-)f=-1;c=getchar();} while(c>=0&&c<=9){x=x*10+c-0;c=getchar();} return x*f; } int N,A,B,C; int a[MAXN]; int pre(int x) { return x==1?N:x-1; } int main() { #ifdef WIN32 freopen("a.in","r",stdin); #endif N=read(),A=read(),B=read(),C=read(),a[1]=read(); for (int i=2;i<=N;i++) a[i] = ((long long)a[i-1] * A + B) % 100000001; for (int i=1;i<=N;i++) a[i] = a[i] % C + 1; double ans=0; for(int i=1;i<=N;i++) ans+=(double)min((double)1/a[i],(double)1/a[pre(i)]); printf("%.3lf",ans); return 0; }

BZOJ2134: 單選錯位(期望亂搞)