题目描述
输入
输出
样例输入
4 0
0 0 1 1
样例输出
512
题解
数论+期望dp,考场上唯一A了的一道题
首先解决正常游戏的操作次数。
易知每个开关都不能被其它的开关组所替代,且每个开关只会影响它和编号比它小的灯。
于是可以从大到小循环一遍,如果一个灯是亮着的,那么把它关闭,把它约数的状态反转,并把num++。
即最终有num个正确选择。
然后解决期望次数。
设b[i]表示从有i个正确选择变为有i-1个正确选择的期望操作次数。
那么可以推出b[i]=i/n*1+(1-i/n)*(1+b[i+1]+b[i]),即b[i]=(b[i+1]*(n-i)+n)/i。
特殊的,b[n+1]=0
然后就可以推出b数组,再判断一下num与k的大小关系并累加一下,最后乘一下n!即可。
考场源代码:
#include <cstdio> #define mod 100003 typedef long long ll; int v[100010]; ll b[100010]; ll qpow(ll x , ll y) { ll ans = 1; while(y) { if(y & 1) ans = ans * x % mod; x = x * x % mod; y >>= 1; } return ans; } int main() { int n , k , i , j , num = 0; ll t = 0; scanf("%d%d" , &n , &k); for(i = 1 ; i <= n ; i ++ ) scanf("%d" , &v[i]); for(i = n ; i >= 1 ; i -- ) { if(v[i]) { for(j = 1 ; j * j <= i ; j ++ ) { if(i % j == 0) { v[j] ^= 1; if(j * j != i) v[i / j] ^= 1; } } num ++ ; } } for(i = n ; i >= 1 ; i -- ) b[i] = (b[i + 1] * (n - i) % mod + n) % mod * qpow(i , mod - 2) % mod; if(n == k || k > num) t = num; else { for(i = num ; i > k ; i -- ) t = (t + b[i]) % mod; t = (t + k) % mod; } for(i = 1 ; i <= n ; i ++ ) t = t * i % mod; printf("%lld\n" , t); return 0; }
【bzoj4872】[Shoi2017]分手是祝愿 数论+期望dp
原文:http://www.cnblogs.com/GXZlegend/p/6764969.html