LINK:小B的夏令营



这道题是以前从没见过的优化dp的方法 不过也在情理之中.

注意读题 千万不要像我这个sb一样 考完连题意都不知道是啥.

一个长方形 要求从上到下联通的概率。

容易发现 K天只是用来计算概率的 和 dp的状态无关。

我们可以逐行 dp.

容易设f[i][l][r]表示前i行 当前行l~r没有被摧毁的概率。

考虑在k天之后第i行 l~r没被摧毁的概率。

l-1在这k天被摧毁了 那么因为有序 概率为\(C(k,l-1)p^{l-1}(1-p)^{k-l+1}\)

对于r的那边同理。

暴力转移 就是枚举上一层的L,R 使得L~R与l~r的交集不为空.

整个时间复杂度为n^5.

考虑 快速得到L~R 对 l~r的贡献 发现 求和l~r有交比较困难。

考虑容斥 那么就是总-左端点在r之后-右端点在l之前即可。

这样就可以优化到n^3 值得注意的是 上述两种情况其实是对等的 所以只需要求一种即可。

inline int mul(int a,int b){return (ll)a*b%mod;}
inline int add(int a,int b){return a+b>=mod?a+b-mod:a+b;}
inline int mux(int a,int b){return a-b<0?a-b+mod:a-b;}
const int MAXN=510,maxn=100010;
int n,m,p,k;
int f[2][MAXN][MAXN];
int fac[maxn],inv[maxn],w[maxn],w1[MAXN];
inline int ksm(int b,int p)
{
int cnt=1;
while(p){if(p&1)cnt=mul(cnt,b);b=mul(b,b);p=p>>1;}
return cnt;
}
inline void prepare(int maxx)
{
fac[0]=1;
rep(1,maxx,i)fac[i]=mul(fac[i-1],i);
inv[maxx]=ksm(fac[maxx],mod-2);
fep(maxx-1,0,i)inv[i]=mul(inv[i+1],i+1);
}
inline int C(int a,int b){return a<b?0:(ll)fac[a]*inv[b]%mod*inv[a-b]%mod;}
int main()
{
freopen("b.in","r",stdin);
freopen("b.out","w",stdout);
get(n);get(m);p=(ll)read()*ksm(read(),mod-2)%mod;get(k);
prepare(max(n,k));
rep(0,min(k,m),i)w[i]=mul(mul(C(k,i),ksm(p,i)),ksm(1-p+mod,k-i));
int u=0;f[u][1][m]=1;
rep(1,n,T)
{
int ans=0;
rep(1,m,i)w1[i]=0;
rep(1,m,i)rep(i,m,j)
{
ans=add(ans,f[u][i][j]);
w1[j]=add(w1[j],f[u][i][j]);
}
rep(1,m,i)w1[i]=add(w1[i],w1[i-1]);
u=u^1;
rep(1,m,i)rep(i,m,j)
f[u][i][j]=mul(mux(mux(ans,w1[i-1]),w1[m-j]),mul(w[i-1],w[m-j]));
}
int ans=0;
rep(1,m,i)rep(i,m,j)ans=add(ans,f[u][i][j]);
put(ans);
return 0;
}

考虑100分的做法:

状态的维度都是n^3的 必须要降维.

按行划分是必要的 所以只能把左端点给省掉或者把右端点给省掉.

我是将前者省掉 设状态f[i][j]表示到了第i行 以j为右端点的概率。

这次需要先枚举l进行转移 实际上这个枚举可以经过预处理之后O(1)得到。

所以 复杂度被降到了nm.

值得注意的是 需要把状态转移方程给写清楚 然后考虑预处理哪些项来计算。

不要像我一样迷瞪半天...

inline int mul(int a,int b){return (ll)a*b%mod;}
inline int add(int a,int b){return a+b>=mod?a+b-mod:a+b;}
inline int mux(int a,int b){return a-b<0?a-b+mod:a-b;}
const int MAXN=1510,maxn=100010;
int n,m,p,k;
int f[MAXN][MAXN];//f[i][j]表示前i行 右端点在j的概率.
int fac[maxn],inv[maxn],w[maxn],w1[maxn],w2[maxn];
inline int ksm(int b,int p)
{
int cnt=1;
while(p){if(p&1)cnt=mul(cnt,b);b=mul(b,b);p=p>>1;}
return cnt;
}
inline void prepare(int maxx)
{
fac[0]=1;
rep(1,maxx,i)fac[i]=mul(fac[i-1],i);
inv[maxx]=ksm(fac[maxx],mod-2);
fep(maxx-1,0,i)inv[i]=mul(inv[i+1],i+1);
}
inline int C(int a,int b){return a<b?0:(ll)fac[a]*inv[b]%mod*inv[a-b]%mod;}
int main()
{
freopen("b.in","r",stdin);
freopen("b.out","w",stdout);
get(n);get(m);p=(ll)read()*ksm(read(),mod-2)%mod;get(k);
prepare(max(n,k));
rep(0,min(k,m),i)w[i]=mul(mul(C(k,i),ksm(p,i)),ksm(1-p+mod,k-i));
f[0][m]=1;
rep(1,n,i)
{
int ww1=0;int cnt=0;int cc=0;
rep(1,m,j)
{
cnt=add(cnt,f[i-1][j]);
ww1=add(ww1,f[i-1][j]);
w1[j]=add(w1[j-1],mul(w[j],ww1));
w2[j]=ww1;
}
rep(1,m,j)
{
cc=add(cc,w[j-1]);
f[i][j]=mux(mux(mul(cc,cnt),w1[j-1]),mul(cc,w2[m-j]));
f[i][j]=mul(f[i][j],w[m-j]);
}
}
int ans=0;
rep(1,m,i)ans=add(ans,f[n][i]);
put(ans);
return 0;
}

5.19 省选模拟赛 小B的夏令营 概率 dp 前缀和优化dp的更多相关文章

  1. 5.19 省选模拟赛 小B的图 最小生成树 LCT

    LINK:小B的图 这道题就比较容易了. 容易想到将询问离线 然后 从小到大排序 那么显然是优先放正图(x+k)的边. 考虑随着x的增大 那么负图上的边会逐渐加进来 一条边被加进来当且仅当 其权值小于 ...

  2. 5.19 省选模拟赛 T1 小B的棋盘 双指针 性质

    LINK:小B的棋盘 考试的时候没有认真的思考 导致没做出来. 容易发现 当k>=n的时候存在无限解 其余都存在有限解 对于30分 容易想到暴力枚举 对称中心 然后 n^2判断. 对于前者 容易 ...

  3. 「HGOI#2019.4.19省选模拟赛」赛后总结

    t1-Painting 这道题目比较简单,但是我比较弱就只是写了一个链表合并和区间DP. 别人的贪心吊打我的DP,嘤嘤嘤. #include <bits/stdc++.h> #define ...

  4. 6.10 省选模拟赛 小C的利是 高斯消元 矩阵行列式

    LINK:小C的利是 想起来把这道题的题解写了 .一个常识:利是在广东那边叫做红包. 关于行列式的题目 不过我不太会23333..口胡还是可以的. 容易想到10分的状压.不过没什么意思. 仔细观察要求 ...

  5. 【洛谷比赛】[LnOI2019]长脖子鹿省选模拟赛 T1 题解

    今天是[LnOI2019]长脖子鹿省选模拟赛的时间,小编表示考的不怎么样,改了半天也只会改第一题,那也先呈上题解吧. T1:P5248 [LnOI2019SP]快速多项式变换(FPT) 一看这题就很手 ...

  6. LOJ 6089 小Y的背包计数问题 —— 前缀和优化DP

    题目:https://loj.ac/problem/6089 对于 i <= √n ,设 f[i][j] 表示前 i 种,体积为 j 的方案数,那么 f[i][j] = ∑(1 <= k ...

  7. Java 第十一届 蓝桥杯 省模拟赛 小明的城堡

    小明用积木搭了一个城堡. 为了方便,小明在搭的时候用的是一样大小的正方体积本,搭在了一个 n 行 m 列的方格图上,每个积木正好占据方格图的一个小方格. 当然,小明的城堡并不是平面的,而是立体的.小明 ...

  8. Java 第十一届 蓝桥杯 省模拟赛 小明植树(DFS)

    小明植树 题目 问题描述 小明和朋友们一起去郊外植树,他们带了一些在自己实验室精心研究出的小树苗. 小明和朋友们一共有 n 个人,他们经过精心挑选,在一块空地上每个人挑选了一个适合植树的位置,总共 n ...

  9. 省选模拟赛第四轮 B——O(n^4)->O(n^3)->O(n^2)

    一 稍微转化一下,就是找所有和原树差距不超过k的不同构树的个数 一个挺trick的想法是: 由于矩阵树定理的行列式的值是把邻接矩阵数值看做边权的图的所有生成树的边权乘积之和 那么如果把不存在于原树中的 ...

随机推荐

  1. SpringBoot + Vue + ElementUI 实现后台管理系统模板 -- 后端篇(一): 搭建基本环境、整合 Swagger、MyBatisPlus、JSR303 以及国际化操作

    相关 (1) 相关博文地址: SpringBoot + Vue + ElementUI 实现后台管理系统模板 -- 前端篇(一):搭建基本环境:https://www.cnblogs.com/l-y- ...

  2. 开源导入导出库Magicodes.IE 多sheet导入教程

    原文作者:tanyongzheng 多Sheet导入教程 说明 本教程主要说明如何使用Magicodes.IE.Excel完成多个Sheet数据的Excel导入. 要点 多个相同格式的Sheet数据导 ...

  3. day59 pip安装django出错解决方案

    在虚拟环境下,输入 pipinstall django ==2.2,安装django,可能会出现超时问题 ​ 这里的报错是网络问题,解决方案有如下三种 (1)多试几次,网络好就装上了 (2)Cmd输入 ...

  4. drf权限,频率,过滤,排序,异常处理

    目录 一.权限 1 权限源码分析 2 自定义权限类 3 内置权限类 二.频率 1 内置频率设置 三.过滤 四.排序 五.异常处理 一.权限 1 权限源码分析 # APIView---->disp ...

  5. Scrapy(五):CrawlSpider的使用

    Scrapy(五):CrawlSpider的使用 说明 :CrawlSpider,就是一个类,是Spider的一个子类,也是一个官方类,因为是子类,所以功能更加的强大,多了一项功能:去指定的页面中来抓 ...

  6. python 装饰器(七):装饰器实例(四)类装饰器装饰类以及类方法

    类装饰器装饰类方法 不带参数 from functools import wraps import types class CatchException: def __init__(self,orig ...

  7. C#-性能-二维数组和数组的数组的性能比较

    两者是3:2的消耗比例 const int NUM = 10000; int[,] vec = new int[NUM, NUM]; Stopwatch sw = Stopwatch.StartNew ...

  8. 采用 Unicode 标点属性方式的正则表达式,可以去掉所有的标点符号,

    public class Test { public static void main(String[] args) { String str = "!!!??!!!!%*)%¥!KTV去符 ...

  9. Python Ethical Hacking - MAC Address & How to Change(3)

    SIMPLE ALGORITHM Goal  -> Check if MAC address was changed. Steps: 1. Execute and read ifconfig. ...

  10. echarts 踩坑 : id必须不同

    我们可能用react前端框架开发项目. 也就是组件化开发. 一个页面里可能有很多组件. 而echarts是寻找特定ID的DOM去渲染的. 也就是说,如果整个页面.包括所有页面组件,有id相同的DOM, ...