Description

Zeit und Raum trennen dich und mich.

时空将你我分开。

\(B\) 君在玩一个游戏,这个游戏由 \(n\) 个灯和 \(n\) 个开关组成,给定这 \(n\) 个灯的初始状态,下标为

从 1 到 \(n\) 的正整数。每个灯有两个状态亮和灭,我们用 1 来表示这个灯是亮的,用 0 表示这个灯是灭的,游戏

的目标是使所有灯都灭掉。但是当操作第 \(i\) 个开关时,所有编号为 \(i\) 的约数(包括 1 和 \(i\))的灯的状态都会被

改变,即从亮变成灭,或者是从灭变成亮。\(B\) 君发现这个游戏很难,于是想到了这样的一个策略,每次等概率随机

操作一个开关,直到所有灯都灭掉。这个策略需要的操作次数很多, \(B\) 君想到这样的一个优化。如果当前局面,

可以通过操作小于等于 \(k\) 个开关使所有灯都灭掉,那么他将不再随机,直接选择操作次数最小的操作方法(这个

策略显然小于等于 \(k\) 步)操作这些开关。\(B\) 君想知道按照这个策略(也就是先随机操作,最后小于等于 \(k\) 步,使

用操作次数最小的操作方法)的操作次数的期望。这个期望可能很大,但是 \(B\) 君发现这个期望乘以 \(n\) 的阶乘一定

是整数,所以他只需要知道这个整数对 100003 取模之后的结果。

Input

第一行两个整数 \(n, k\)。

接下来一行 \(n\) 个整数,每个整数是 0 或者 1,其中第 \(i\) 个整数表示第 \(i\) 个灯的初始情况。

\(1 \leq n \leq 100000, 0 \leq k \leq n\);

Output

输出一行,为操作次数的期望乘以 \(n\) 的阶乘对 100003 取模之后的结果。

Sample Input

4 0

0 0 1 1

Sample Output

512


想法

又是个小文艺题,原谅我有点激动。。。

首先从 \(n\) 到 \(1\) 扫一遍,哪些灯需要被按1次是可以知道的,而且必须按这些灯。

如果不小心按了其他的灯,就必须再按一次变回来。

神奇的 \(dp\) 方式,设 \(f[i]\) 表示从最少需要按 \(i\) 次的状态变到最少需要按 \(i-1\) 次的状态的期望步数。

随机按一次,按中 \(i\) 个需要按的键之一的概率是 \(\frac{i}{n}\) ,按后变到了至少按 \(i-1\) 次的状态,步数为1

若没有按中需要的键,概率是 \(\frac{n-i}{n}\) , 此时至少需要按 \(i+1\) 个键,应再按 \(f[i+1]\) 次恢复至少按 \(i\) 次的状态,再按 \(f[i]\) 次到至少 \(i-1\) 次的状态,总共步数为 \(1+f[i+1]+f[i]\)

综上所述,可列出式子

\(
f[i]=\frac{i}{n}+\frac{n-i}{n}(1+f[i+1]+f[i])
\)

合并同类项,整理一下得出

\(
f[i]=\frac{n+(n-i)f[i+1]}{i}
\)

边界条件 \(f[n]=1\)

从 \(n\) 往前求 \(f[]\) ,一直求到 \(f[k+1]\)

扫一遍得出原状态至少按的次数 \(t\)

若 \(t \leq k\) ,则总期望值就是 \(t\) ; 否则是 \(k+\sum\limits_{i=k+1}^t f[i]\)

最后别忘了乘以 \(n\) 的阶乘!


代码

#include<cstdio>
#include<iostream>
#include<algorithm> #define xzy 100003 using namespace std; const int N = 100005;
typedef long long ll; int n,k,t;
int a[N]; int Pow_mod(int x,int y){
int ret=1;
while(y){
if(y&1) ret=((ll)ret*x)%xzy;
x=((ll)x*x)%xzy;
y>>=1;
}
return ret;
}
int f[N],inv[N]; int main()
{
scanf("%d%d",&n,&k);
for(int i=1;i<=n;i++) scanf("%d",&a[i]); t=0;
for(int i=n;i>0;i--){
if(!a[i]) continue;
t++;
for(int j=1;j*j<=i;j++)
if(i%j==0){
a[j]^=1;
if(j*j!=i) a[i/j]^=1;
}
} inv[1]=1;
for(int i=2;i<=n;i++) inv[i]=xzy-1ll*(xzy/i)*inv[xzy%i]%xzy; int ans=k;
f[n]=1;
for(int i=n-1;i>k;i--)
f[i]=(1ll*(n-i)*f[i+1]%xzy+n)%xzy*inv[i]%xzy; if(t<=k) ans=t;
else for(int i=k+1;i<=t;i++) ans=(ans+f[i])%xzy;
for(int i=2;i<=n;i++) ans=((ll)ans*i)%xzy;
printf("%d\n",ans); return 0;
}

[bzoj4872] [洛谷P3750] [六省联考2017] 分手是祝愿的更多相关文章

  1. 洛谷P3750 [六省联考2017]分手是祝愿(期望dp)

    传送门 嗯……概率期望这东西太神了…… 先考虑一下最佳方案,肯定是从大到小亮的就灭(这个仔细想一想应该就能发现) 那么直接一遍枚举就能$O(nlogn)$把这个东西给搞出来 然后考虑期望dp,设$f[ ...

  2. 洛谷 P3750 [六省联考2017]分手是祝愿

    传送门 题解 //Achen #include<algorithm> #include<iostream> #include<cstring> #include&l ...

  3. 洛谷 P3750 - [六省联考2017]分手是祝愿(期望 dp)

    题面传送门 首先我们需注意到这样一个性质:那就是对于任何一种状态,将其变为全 \(0\) 所用的最小步数的方案是唯一的--考虑编号为 \(n\) 的灯,显然如果它原本是暗着的就不用管它了,如果它是亮着 ...

  4. BZOJ 4872 luogu P3750 [六省联考2017]分手是祝愿

    4872: [Shoi2017]分手是祝愿 Time Limit: 20 Sec  Memory Limit: 512 MB[Submit][Status][Discuss] Description ...

  5. P3750 [六省联考2017]分手是祝愿 期望DP

    \(\color{#0066ff}{ 题目描述 }\) Zeit und Raum trennen dich und mich. 时空将你我分开. B 君在玩一个游戏,这个游戏由 \(n\) 个灯和 ...

  6. luogu P3750 [六省联考2017]分手是祝愿

    luogu loj 可以发现在最优策略中,每种操作最多只会做一次,并且操作的先后顺序并不会影响答案,所以考虑从后往前扫,碰到一个\(1\)就对这个位置\(i\)进行操作,这样的操作一定是最优策略.记最 ...

  7. 洛谷 P3747 [六省联考2017]相逢是问候 解题报告

    P3747 [六省联考2017]相逢是问候 题目描述 \(\text {Informatik verbindet dich und mich.}\) 信息将你我连结. \(B\) 君希望以维护一个长度 ...

  8. [BZOJ4872][六省联考2017]分手是祝愿(期望DP)

    4872: [Shoi2017]分手是祝愿 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 516  Solved: 342[Submit][Statu ...

  9. bzoj千题计划266:bzoj4872: [六省联考2017]分手是祝愿

    http://www.lydsy.com/JudgeOnline/problem.php?id=4872 一种最优解是 从大到小灯有亮的就灭掉 最优解是唯一的,且关灯的顺序没有影响 最优解 对每个开关 ...

随机推荐

  1. javascript中的深拷贝与浅拷贝

    javascript中的深拷贝与浅拷贝 基础概念 在了解深拷贝与浅拷贝的时候需要先了解一些基础知识 核心知识点之 堆与栈 栈(stack)为自动分配的内存空间,它由系统自动释放: 堆(heap)则是动 ...

  2. vue-learning:12-vue获取模板内容的方式

    vue获取模板内容的方式 目录 outerHTML获取内容 template属性获取内容 ES6的字符串模板 <template>标签 <srcipt type="text ...

  3. 2018-4-12-win10-uwp-使用油墨输入

    title author date CreateTime categories win10 uwp 使用油墨输入 lindexi 2018-04-12 14:19:58 +0800 2018-2-13 ...

  4. MFC防止进程重复建立

    原文:https://blog.csdn.net/zhang11wu4/article/details/7100839 在APP类的InitInstance()的最前面加入以下代码,建立互斥区,可防止 ...

  5. Visual Studio插件【一】:前端

    JQuery Code Snippets https://github.com/kspearrin/Visual-Studio-jQuery-Code-Snippets 简单用法 jq   +tab ...

  6. springmvc整合freemarker教程(转)

    1.介绍 我最近喜欢freemarker在网上找了大半天.都没有找到一个简单又容易理解的案例.虽然只是一个模板技术.但是相对刚开始什么都不知道的,很难入手.下面是自学(其实是谷歌和百度的东找西补).写 ...

  7. JavaScript数组的方法 | 学习笔记分享

    数组 数组的四个常用方法 push() 该方法可以向数组的末尾添加一个或多个元素,并返回数组的新长度 可以将要添加的元素作为方法的参数传递,这些元素将会自动添加到数组的末尾 pop() 该方法可以删除 ...

  8. 机器学习算法概述第五章——CART算法

    特点: 是一个二叉树,元素可以重复利用,可以做回归也可以做分类,分类用最小二乘法,即误差平方和最小 切割方法: 对于可量化的x来说: 切割点通常为两个x的平均值 左右两部分分别取均值,再评判以哪个分割 ...

  9. .Net快速上手Nlog日志组件

    目录 一.NLog 简介 二.NLog 安装 三. NLog 配置 四.程序代码中写日志 五.参考 一.NLog 简介 NLog是适用于各种.NET平台的灵活,免费的日志记录平台.NLog使写入多个目 ...

  10. 使用vue 对二进制文件 实现下载(blob对象

    有很多网站会涉及到文件下载,这里我们使用axios 发送请求 接受数据 第一步 模仿jQ 封装接口 Vue.prototype.$xlsx_post = function (url, data, fu ...