DP式子比后面的东西难推多了

LOJ2304

Luogu P3824

UOJ #316


题意

给定一个长度为$ 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」泳池的更多相关文章

  1. LOJ#2304. 「NOI2017」泳池

    $n \leq 1e9$底边长的泳池,好懒啊泥萌自己看题吧,$k \leq 1000$.答案对998244353取膜. 现在令$P$为安全,$Q$为危险的概率.刚好$K$是极其不好算的,于是来算$\l ...

  2. LOJ 2304 「NOI2017」泳池——思路+DP+常系数线性齐次递推

    题目:https://loj.ac/problem/2304 看了各种题解…… \( dp[i][j] \) 表示有 i 列.第 j 行及以下默认合法,第 j+1 行至少有一个非法格子的概率,满足最大 ...

  3. LOJ_2305_「NOI2017」游戏 _2-sat

    LOJ_2305_「NOI2017」游戏 _2-sat 题意: 给你一个长度为n的字符串S,其中第i个字符为a表示第i个地图只能用B,C两种赛车,为b表示第i个地图只能用A,C两种赛车,为c表示第i个 ...

  4. 「NOI2017」游戏

    「NOI2017」游戏 题目描述 小 L 计划进行 \(n\) 场游戏,每场游戏使用一张地图,小 L 会选择一辆车在该地图上完成游戏. 小 L 的赛车有三辆,分别用大写字母 \(A\).\(B\).\ ...

  5. loj #2305. 「NOI2017」游戏

    #2305. 「NOI2017」游戏 题目描述 小 L 计划进行 nnn 场游戏,每场游戏使用一张地图,小 L 会选择一辆车在该地图上完成游戏. 小 L 的赛车有三辆,分别用大写字母 AAA.BBB. ...

  6. LOJ2305 「NOI2017」游戏

    「NOI2017」游戏 题目背景 狂野飙车是小 L 最喜欢的游戏.与其他业余玩家不同的是,小 L 在玩游戏之余,还精于研究游戏的设计,因此他有着与众不同的游戏策略. 题目描述 小 L 计划进行$n$场 ...

  7. LOJ2303 「NOI2017」蚯蚓排队

    「NOI2017」蚯蚓排队 题目描述 蚯蚓幼儿园有$n$只蚯蚓.幼儿园园长神刀手为了管理方便,时常让这些蚯蚓们列队表演. 所有蚯蚓用从$1$到$n$的连续正整数编号.每只蚯蚓的长度可以用一个正整数表示 ...

  8. LibreOJ2302 - 「NOI2017」整数

    Portal Description 有一个整数\(x=0\),对其进行\(n(n\leq10^6)\)次操作: 给出\(a(|a|\leq10^9),b(b\leq30n)\),将\(x\)加上\( ...

  9. 「NOI2017」蔬菜 解题报告

    「NOI2017」蔬菜 首先考虑流 可以从 \(s\) 流入表示得到蔬菜,流出到 \(t\) 表示卖出蔬菜,给每个蔬菜拆点,并给它它每天应得的蔬菜. 但是我们没办法直接给,注意到如果把变质看成得到并可 ...

随机推荐

  1. 无post按钮提交表单

    <form id="form1" name="form" action="url" method="GET"> ...

  2. mmap:速度快+整块操作

    mmap使得可以将设备内存映射到用户空间,从而使得用户程序获得访问硬件的能力,mmap的动作需要由内核中的驱动来实现.在使用mmap映射后,用户程序对给定范围的内存的读写就变成了对设备内存的读写,也就 ...

  3. YAML详解

    1   YAML简介 YAML,即YAML Ain’t Markup Language的缩写,YAML 是一种简洁的非标记语言.YAML以数据为中心,使用空白,缩进,分行组织数据,从而使得表示更加简洁 ...

  4. ✔ OI Diary ★

    一 | 2019-3-28 1.整晨,云之考矣,暴后皆不会,邃无感而写斯普雷尔,然则午后知暴可六十哉. 然则斯普雷毙,虽特判之矣,然则暴只判二十哉,呜呼! ​2.午间归宿,视白购书一本,目触,感之甚集 ...

  5. cmd执行超大sql文件

    osql -S 127.0.0.1 -U sa -P 123456 -i d:\test.sql osql为SQL Server的命令,要在cmd中执行该命令,一般安装完SQL Server后该命令对 ...

  6. MyBatis 传List参数 nested exception is org.apache.ibatis.binding.BindingException: Parameter 'idList' not found.

    在MyBatis传入List参数时,MyBatis报错:nested exception is org.apache.ibatis.binding.BindingException: Paramete ...

  7. 【ubuntu】Ubuntu 修改 Apache2 运行用户/用户组及修改方法

    我们在安装apache后,有时在上传文件的时候,提示没有权限或者是不可写,我们都会去查文件夹的权限.通过ls -l /var/www/html/website可以很直观的看出我们文件和文件夹的权限,d ...

  8. Android——图片轮播

    Android技术——轮播功能 轮播需要什么? 答:实现图片与广告语展示.循环播发以及手动切换.支持加载本地与网络图片. 性能优化? 答:多张图片与指示器展示.自动与定时.循环播发.滑动流畅并且无卡顿 ...

  9. LODOP设置超文本不自动分页的方法

    在LODOP中,超文本超过打印项高度会自动分页,自动分页有两种情况:超过设置的打印项高度,超过纸张.这里是指高度,超过纸张宽度的超文本不会显示,会隐藏掉. 如果你不了解什么是LODOP中的超文本打印项 ...

  10. D. Maximum Diameter Graph 贪心+图论+模拟

    题意:给出n个点的度数列 上限(实际点可以小于该度数列)问可以构造简单路最大长度是多少(n个点要连通 不能有平行边.重边) 思路:直接构造一条长链  先把度数为1的点 和度数大于1的点分开  先把度数 ...