「NOI2017」泳池
DP式子比后面的东西难推多了
题意
给定一个长度为$ n$高为$ \infty$的矩形
每个点有$ 1-P$的概率不可被选择
求最大的和底边重合的不包含不可选点的矩形的面积为$ K$的概率
$ n \leq 10^9 k \leq 10^3$
题解
K可以出到50000的
首先考虑DP
面积恰好为$ K$的概率可以差分为不高于$ K$的概率减去不高于$ K-1$的概率
设$ f[i][j]$表示长度为$ i$的矩形,从底边起$ j$行都可选,最大面积不大于$ K$的概率
边界为$ f[0][j]=1,f[i][j]=0 当且仅当i*j>k$
考虑转移,要么第$ j+1$行也都可选,要么第$ j+1$行有不可选的位置
对于第二种情况我们枚举从左到右第一个不可选的位置
有转移方程式
$$f[i][j]=f[i][j+1]*P^i+\sum_{k=1}^iP^{k-1}(1-P)f[k-1][j+1]·f[i-k][j]$$
我们要求的是$ f[n][0]$
由于$ i*j \leq K$因此复杂度大致是$ n·k \log k$的
可以得$ 70$分
容易发现当$ n$远大于$ k$的时候,每连续$ k$列必然有一列最低端有不可选点
令$ F[i]$表示当$ i>k$时,长度为$ i$的矩形的答案
枚举从右往左第一个不可选点,有转移方程式
$$ F[i]=f[i-k]*(1-P)*f[k-1][1]*P^{k-1}$$
这是一个线性递推的标准形式,可以用特征多项式优化到$ O(k^2 \log n)$甚至$ O(k \log k \log n)$
如果采用后一种的话复杂度的瓶颈在于前面的$ O(k^2)DP$,这部分可以用分治$ NTT$优化
然而我并没有写
代码
- #include<ctime>
- #include<cmath>
- #include<cstdio>
- #include<cstring>
- #include<iostream>
- #include<algorithm>
- #include<queue>
- #include<vector>
- #define p 998244353
- #define rt register int
- #define ll long long
- using namespace std;
- inline ll read(){
- ll x=;char zf=;char ch=getchar();
- while(ch!='-'&&!isdigit(ch))ch=getchar();
- if(ch=='-')zf=-,ch=getchar();
- while(isdigit(ch))x=x*+ch-'',ch=getchar();return x*zf;
- }
- void write(ll y){if(y<)putchar('-'),y=-y;if(y>)write(y/);putchar(y%+);}
- void writeln(const ll y){write(y);putchar('\n');}
- namespace poly{
- vector<int>R;
- int ksm(int x,int y=p-){
- int ans=;
- for(rt i=y;i;i>>=,x=1ll*x*x%p)if(i&)ans=1ll*ans*x%p;
- return ans;
- }
- void NTT(int n,vector<int>&A,int fla){
- A.resize(n);
- for(rt i=;i<n;i++)if(i>R[i])swap(A[i],A[R[i]]);
- for(rt i=;i<n;i<<=){
- int w=ksm(,(p-)//i);
- for(rt j=;j<n;j+=i<<){
- int K=;
- for(rt k=;k<i;k++,K=1ll*K*w%p){
- int x=A[j+k],y=1ll*K*A[i+j+k]%p;
- A[j+k]=(x+y)%p,A[i+j+k]=(x-y)%p;
- }
- }
- }
- if(fla==-){
- reverse(A.begin()+,A.end());
- int invn=ksm(n);
- for(rt i=;i<n;i++)A[i]=1ll*A[i]*invn%p;
- }
- }
- vector<int>Mul(vector<int>x,vector<int>y){
- int lim=,sz=x.size()+y.size()-;
- while(lim<=sz)lim<<=;R.resize(lim);
- for(rt i=;i<lim;i++)R[i]=(R[i>>]>>)|(i&)*(lim>>);
- NTT(lim,x,);NTT(lim,y,);
- for(rt i=;i<lim;i++)x[i]=1ll*x[i]*y[i]%p;
- NTT(lim,x,-);x.resize(sz);
- return x;
- }
- vector<int>sqr(vector<int>x){
- int lim=,sz=x.size()*-;
- while(lim<=sz)lim<<=;R.resize(lim);
- for(rt i=;i<lim;i++)R[i]=(R[i>>]>>)|(i&)*(lim>>);
- NTT(lim,x,);for(rt i=;i<lim;i++)x[i]=1ll*x[i]*x[i]%p;
- NTT(lim,x,-);x.resize(sz);
- return x;
- }
- vector<int>Inv(vector<int>A,int n=-){
- if(n==-)n=A.size();
- if(n==)return vector<int>(,ksm(A[]));
- vector<int>b=Inv(A,(n+)/);
- int lim=;while(lim<=n+n)lim<<=;R.resize(lim);
- for(rt i=;i<lim;i++)R[i]=(R[i>>]>>)|(i&)*(lim>>);
- A.resize(n);NTT(lim,A,);NTT(lim,b,);
- for(rt i=;i<lim;i++)A[i]=1ll*b[i]*(2ll-1ll*A[i]*b[i]%p)%p;
- NTT(lim,A,-);A.resize(n);
- return A;
- }
- vector<int>Div(vector<int>A,vector<int>B){
- int n=A.size(),m=B.size();
- reverse(A.begin(),A.end());
- reverse(B.begin(),B.end());
- A.resize(n-m+),B.resize(n-m+);
- int lim=;while(lim<=*(n-m+))lim<<=;R.resize(lim);
- for(rt i=;i<lim;i++)R[i]=(R[i>>]>>)|(i&)*(lim>>);
- vector<int>ans=Mul(A,Inv(B));ans.resize(n-m+);
- reverse(ans.begin(),ans.end());
- return ans;
- }
- vector<int>add(vector<int>A,vector<int>B){
- int len=max(A.size(),B.size());A.resize(len+);
- for(rt i=;i<=len;i++)(A[i]+=B[i])%=p;
- return A;
- }
- vector<int>sub(vector<int>A,vector<int>B){
- int len=max(A.size(),B.size());A.resize(len+);
- for(rt i=;i<=len;i++)(A[i]-=B[i])%=p;
- return A;
- }
- vector<int>Mod(vector<int>x,vector<int>y){
- if(x.size()<=y.size())return x;
- vector<int>ans=Div(x,y);
- ans=sub(x,Mul(y,ans));
- while(!ans[ans.size()-])ans.pop_back();
- if(ans.size()>y.size())ans.resize(y.size());
- return ans;
- }
- }
- using namespace poly;
- int a[];
- vector<int>fmo;
- vector<int>ksm(vector<int>x,int y){
- if(y==)return x;
- vector<int>ans=Mod(sqr(ksm(x,y>>)),fmo);
- if(y&){
- ans.push_back();
- for(rt i=ans.size()-;i>=;i--)ans[i+]=ans[i],ans[i]=;
- }
- return ans;
- }
- using namespace poly;
- int k,m,n,x,y,z,cnt,ans,K,P;
- int f[][],mi[];
- //f[i][j]长度为i合法高度最低至少为j的合法概率
- void calc(int n,int K,int fla){
- memset(f,,sizeof(f));
- for(rt i=;i<=K+;i++)f[][i]=;
- for(rt i=;i<=K+;i++)
- for(rt j=K/i;j>=;j--){
- f[i][j]=1ll*f[i][j+]*mi[i]%p;
- for(rt k=;k<=i;k++)(f[i][j]+=1ll*f[k-][j+]*mi[k-]%p*(+p-P)%p*f[i-k][j]%p)%=p;
- }
- fmo.resize(K+);
- for(rt j=;j<=K+;j++)fmo[K+-j]=-1ll*mi[j-]*f[j-][]%p*(p+-P)%p;
- fmo[K+]=;int ret=;
- if(n<=K+)ret=f[n][];else {
- vector<int>x;x.push_back();x.push_back();
- x=ksm(x,n);
- for(rt i=;i<=K+;i++)(ret+=1ll*f[i][]*x[i]%p)%=p;
- }
- (ans+=ret*fla)%=p;
- }
- int main(){
- // file("pool");
- n=read();K=read();x=read();y=read();P=1ll*x*ksm(y)%p;
- mi[]=;for(rt i=;i<=K+;i++)mi[i]=1ll*mi[i-]*P%p;
- calc(n,K,);calc(n,K-,-);cout<<(ans+p)%p;
- return ;
- }
「NOI2017」泳池的更多相关文章
- LOJ#2304. 「NOI2017」泳池
$n \leq 1e9$底边长的泳池,好懒啊泥萌自己看题吧,$k \leq 1000$.答案对998244353取膜. 现在令$P$为安全,$Q$为危险的概率.刚好$K$是极其不好算的,于是来算$\l ...
- LOJ 2304 「NOI2017」泳池——思路+DP+常系数线性齐次递推
题目:https://loj.ac/problem/2304 看了各种题解…… \( dp[i][j] \) 表示有 i 列.第 j 行及以下默认合法,第 j+1 行至少有一个非法格子的概率,满足最大 ...
- LOJ_2305_「NOI2017」游戏 _2-sat
LOJ_2305_「NOI2017」游戏 _2-sat 题意: 给你一个长度为n的字符串S,其中第i个字符为a表示第i个地图只能用B,C两种赛车,为b表示第i个地图只能用A,C两种赛车,为c表示第i个 ...
- 「NOI2017」游戏
「NOI2017」游戏 题目描述 小 L 计划进行 \(n\) 场游戏,每场游戏使用一张地图,小 L 会选择一辆车在该地图上完成游戏. 小 L 的赛车有三辆,分别用大写字母 \(A\).\(B\).\ ...
- loj #2305. 「NOI2017」游戏
#2305. 「NOI2017」游戏 题目描述 小 L 计划进行 nnn 场游戏,每场游戏使用一张地图,小 L 会选择一辆车在该地图上完成游戏. 小 L 的赛车有三辆,分别用大写字母 AAA.BBB. ...
- LOJ2305 「NOI2017」游戏
「NOI2017」游戏 题目背景 狂野飙车是小 L 最喜欢的游戏.与其他业余玩家不同的是,小 L 在玩游戏之余,还精于研究游戏的设计,因此他有着与众不同的游戏策略. 题目描述 小 L 计划进行$n$场 ...
- LOJ2303 「NOI2017」蚯蚓排队
「NOI2017」蚯蚓排队 题目描述 蚯蚓幼儿园有$n$只蚯蚓.幼儿园园长神刀手为了管理方便,时常让这些蚯蚓们列队表演. 所有蚯蚓用从$1$到$n$的连续正整数编号.每只蚯蚓的长度可以用一个正整数表示 ...
- LibreOJ2302 - 「NOI2017」整数
Portal Description 有一个整数\(x=0\),对其进行\(n(n\leq10^6)\)次操作: 给出\(a(|a|\leq10^9),b(b\leq30n)\),将\(x\)加上\( ...
- 「NOI2017」蔬菜 解题报告
「NOI2017」蔬菜 首先考虑流 可以从 \(s\) 流入表示得到蔬菜,流出到 \(t\) 表示卖出蔬菜,给每个蔬菜拆点,并给它它每天应得的蔬菜. 但是我们没办法直接给,注意到如果把变质看成得到并可 ...
随机推荐
- JavaScript-创建日志调试对象(面向对象实例)
参考自http://www.2cto.com/kf/201312/261990.html IC.js文件 自己封装的js类库 /** * * @authors Your Name (you@examp ...
- Java中a+=b和a=a+b的区别
在Java语言中a+=b和a=a+b是有区别的,主要的区别是在运算时精度的问题,当然了-=.*=./=,%=也都是一个道理.这里以a+=b和a=a+b为例做说明. (1)下面以一段Java程序为例,试 ...
- 初识shell编程
1.shell编程之为什么学.怎么学 为什么学shell编程 Linux系统批量管理 提升工作效率,减少重复工作 学好shell编程所需要的基础知识 熟悉使用vim编辑器 熟悉SSH终端 熟练掌握Li ...
- Linux之指令 重定向 文件覆盖>和文件追加>>
指令>和>>区别 指令 > : 如果文件存在,将原来文件的内容覆盖:原文件不存在则创建文件,再添加信息. 指令 >>:不会覆盖原文件内容,将内容追加到文件的尾部. ...
- 转://看懂Oracle中的执行计划
一.什么是Oracle执行计划? 执行计划是一条查询语句在Oracle中的执行过程或访问路径的描述 二.怎样查看Oracle执行计划? 2.1 explain plan for命令查看执行计划 在sq ...
- 第1章 初始Docker容器
1.1 什么是Docker slogan:Build Ship Run Any App Anywher.关键在于Ship,通过把程序和程序运行所需要的环境一起交付. Linux容器技术: Docker ...
- python3 二分法查找
'''二分法查找有序列表掐头去尾取中间查找列表中xx在不在列表中,在,则返回索引值'''# lst = [1, 4, 6, 8, 9, 21, 23, 26, 35, 48, 49, 54, 67, ...
- SpringBoot标准Properties
# =================================================================== # COMMON SPRING BOOT PROPERTIE ...
- C++笔记-数组指针/二维数组转换指针
参考资料: 1. 作者 BensonLaur :https://www.cnblogs.com/BensonLaur/p/6367077.html 2. https://blog.csdn.net/ ...
- 三:OVS+GRE之完整网络流程
知识点一:linux网桥提供安全组 知识点二:每新建一个网络,在网络节点都会新建一个namespace,只要为该网络建立子网,那么该namespace里就新增dhcp来为该子网分配ip,也可以为该网络 ...