UOJ#316. 【NOI2017】泳池
传送门
一道 \(DP\) 好题
设 \(q\) 为一个块合法的概率
套路一恰好为 \(k\) 的概率不好算,算小于等于 \(k\) 的减去小于等于 \(k-1\) 的
那么设 \(f_i\) 表示宽为 \(i\) 的合法的泳池面积都小于等于 \(k\) 的概率
设 \(g_i\) 表示宽为 \(i\) 的合法的泳池面积都小于等于 \(k\) 且最下面一行都合法的概率
那么考虑转移 \(f\)
套路二强制前面的满足一定的性质,后面接一段不满足的
首先 \(f_i+=g_i\),然后枚举放一个不合法的块在最下面
那么贡献就是 \(\sum_{j=0}^{i-1}f_{i-j-1}g_j(1-q)\)
考虑怎么得到 \(g\),考虑到 \(g\) 表示的一定是下面都合法,上面是一个不合法的轮廓线的状态
那么枚举轮廓线的最下面的点
设 \(dp_{i,j}\) 表示宽为 \(i\) 最下面一行都合法,且向上第一个不合法的高度为 \(j+1\) 的概率
仍然用套路二转移
首先 \(i\times j \le k\)
\(dp_{i,j}=(1-q)q^j\sum_{l=1}^{i-1}(\sum_{x\ge j}dp_{l-1,x})(\sum_{x>j}dp_{i-l,x})\)
显然有用的状态只有 \(\Theta(klnk)\),然后前缀和优化做到 \(\Theta(k^2lnk)\)
直接这样子写就有 \(70\) 分
观察这个东西
\(g_i+\sum_{j=0}^{i-1}f_{i-j-1}g_j(1-q)\)
就是个常系数齐次线性递推的形式
套用线性递推即可
# include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn(1005);
const int mod(998244353);
inline void Inc(int &x, int y) {
x = x + y >= mod ? x + y - mod : x + y;
}
inline int Pow(ll x, int y) {
register ll ret = 1;
for (; y; y >>= 1, x = x * x % mod)
if (y & 1) ret = ret * x % mod;
return ret;
}
int f[maxn], g[maxn], dp[maxn][maxn], n, pw[maxn], q;
int p[maxn], tmp[maxn << 1], a[maxn], b[maxn];
inline void Mul(int *x, int *y, int *z, int k) {
register int i, j, inv, t = k << 1;
memset(tmp, 0, sizeof(tmp));
for (i = 0; i < k; ++i)
for (j = 0; j < k; ++j) Inc(tmp[i + j], (ll)x[i] * y[j] % mod);
for (i = t; i >= k; --i)
if (tmp[i])
for (inv = tmp[i], j = 0; j <= k; ++j) Inc(tmp[i - j], mod - (ll)p[k - j] * inv % mod);
for (i = 0; i < k; ++i) z[i] = tmp[i];
}
inline int Solve(int k) {
if (!k) return Pow(q, n);
memset(dp, 0, sizeof(dp)), memset(f, 0, sizeof(f));
register int i, j, l, ans = 0;
for (i = 0; i <= 1001; ++i) dp[0][i] = 1;
for (i = 1; i <= k; ++i)
for (j = k / i; j; --j) {
for (l = 1; l <= i; ++l) Inc(dp[i][j], (ll)dp[l - 1][j] * dp[i - l][j + 1] % mod);
dp[i][j] = (ll)q * pw[j] % mod * dp[i][j] % mod;
Inc(dp[i][j], dp[i][j + 1]);
}
for (i = 0; i <= k; ++i) g[i + 1] = (ll)q * dp[i][1] % mod, f[i] = dp[i][1];
for (i = 1; i <= k; ++i)
for (j = 1, l = min(i, k + 1); j <= l; ++j) Inc(f[i], (ll)f[i - j] * g[j] % mod);
memset(p, 0, sizeof(p)), memset(a, 0, sizeof(a)), memset(b, 0, sizeof(b));
for (p[++k] = 1, i = 0; i < k; ++i) p[i] = mod - g[k - i];
a[0] = b[1] = 1;
for (i = n; i; i >>= 1, Mul(b, b, b, k))
if (i & 1) Mul(a, b, a, k);
for (i = 0; i < k; ++i) Inc(ans, (ll)a[i] * f[i] % mod);
return ans;
}
int k;
int main() {
register int x, y, i;
scanf("%d%d%d%d", &n, &k, &x, &y);
pw[0] = 1, pw[1] = (ll)x * Pow(y, mod - 2) % mod, q = (mod + 1 - pw[1]) % mod;
for (i = 2; i <= 1000; ++i) pw[i] = (ll)pw[i - 1] * pw[1] % mod;
printf("%d\n", (Solve(k) - Solve(k - 1) + mod) % mod);
return 0;
}
UOJ#316. 【NOI2017】泳池的更多相关文章
- [NOI2017]泳池——概率DP+线性递推
[NOI2017]泳池 实在没有思路啊~~~ luogu题解 1.差分,转化成至多k的概率减去至多k-1的概率.这样就不用记录“有没有出现k”这个信息了 2.n是1e9,感觉要递推然后利用数列的加速技 ...
- BZOJ4944: [Noi2017]泳池
BZOJ4944: [Noi2017]泳池 题目背景 久莲是个爱玩的女孩子. 暑假终于到了,久莲决定请她的朋友们来游泳,她打算先在她家的私人海滩外圈一块长方形的海域作为游泳场. 然而大海里有着各种各样 ...
- 【BZOJ4944】[NOI2017]泳池(线性常系数齐次递推,动态规划)
[BZOJ4944][NOI2017]泳池(线性常系数齐次递推,动态规划) 首先恰好为\(k\)很不好算,变为至少或者至多计算然后考虑容斥. 如果是至少的话,我们依然很难处理最大面积这个东西.所以考虑 ...
- UOJ#316. 【NOI2017】泳池 动态规划,Berlekamp-Massey,Cayley-Hamilton定理
原文链接www.cnblogs.com/zhouzhendong/p/UOJ316.html 题解 首先,我们将答案转化成最大矩形大小 \(\leq k\) 的概率 减去 \(\leq k-1\) 的 ...
- BZOJ 4945 UOJ #317 NOI2017 游戏 2-SAT 拓扑排序
http://uoj.ac/problem/317 https://www.lydsy.com/JudgeOnline/problem.php?id=4945 我现在的程序uoj的额外数据通过不了,b ...
- [NOI2017]泳池
题目描述 有一个长为\(n\),高为1001的网格,每个格子有\(p\)的概率为1,\((1-p)\)的概率0,定义一个网格的价值为极大的全一矩形,且这个矩形的底要贴着网格的底,求这个网格的价值为\( ...
- Luogu3824 [NOI2017]泳池 【多项式取模】【递推】【矩阵快速幂】
题目分析: 用数论分块的思想,就会发现其实就是连续一段的长度$i$的高度不能超过$\lfloor \frac{k}{i} \rfloor$,然后我们会发现最长的非$0$一段不会超过$k$,所以我们可以 ...
- 「NOI2017」泳池
DP式子比后面的东西难推多了 LOJ2304 Luogu P3824 UOJ #316 题意 给定一个长度为$ n$高为$ \infty$的矩形 每个点有$ 1-P$的概率不可被选择 求最大的和底边重 ...
- [学习笔记]Cayley-Hilmiton
Cayley–Hamilton theorem - Wikipedia 其实不是理解很透彻,,,先写上 简而言之: 是一个知道递推式,快速求第n项的方法 k比较小的时候可以用矩阵乘法 k是2000,n ...
随机推荐
- 总结day3 ---- 进制转换,字符串切片,字符串常用方法.,for 循环,
前情提要: int 的相关操作 进制转换 bit_lenth() str 的索引,以及常用的相关方法 for 循环 索引 切片 相关方法 一 : int 的相关操作 int 主要用于生活中的计算问题 ...
- Linux Shell编程、变量、控制语句
为什么要学习Shell编程 1)Linux运维工程师在进行服务器集群管理时,需要编写Shell程序来进行服务器管理. 2)对于JavaEE和Python程序员来说,工作的需要,你的老大会要求你编写一些 ...
- GDB:从单线程调试到多线程调试(MFiX单步调试)
GDB:从单线程调试到多线程调试 1. 裸跑GDB 1.1 安装GDB sudo apt-get install gdb 1.2 编译程序 由于需要调试,因此编译的时候需要添加-g编译参数: 1.3 ...
- WPF实现WORD 2013墨迹批注功能
1 前言 WORD 2013可以使用墨迹在文档上面标注,本文讲述通过WPF第三方控件实现类似主要功能如下: 名称 描述 墨迹标注 不论是否触摸屏环境下可以开始墨迹功能,并实现鼠标/触摸在文档任意位置绘 ...
- java翻译到mono C#实现系列(2) mono实现GridView 横向滚动
群里的朋友问GridView 横向滚动怎么实现,我就百度了,参考http://blog.csdn.net/lonely_fireworks/article/details/7841134写了个mono ...
- Jexus进程守护工具jws.guard
一个运行中的进程,难免会因为各种各样的原因无缘无故的宕掉(比如网站瞬间的负载过高.内存不足等),而Jexus宕掉的后果往往只有一个:对外提供服务的网站无法访问了.因此,我们需要最大限度的保障我们的网站 ...
- Asp.NET MVC 拍卖网站,拆解【1】预览与目录
本人最近带创业团队基本做完了一个艺术品拍卖的外包项目,分为网站前台(asp.net mvc5),网站管理员管理的后台使用的9900端口(asp.net mvc5),监听拍卖状态的windows服务,为 ...
- BigDecimal加减乘除运算(转)
java.math.BigDecimal.BigDecimal一共有4个够造方法,让我先来看看其中的两种用法: 第一种:BigDecimal(double val) Translates a doub ...
- java service wrapper将java程序包装成系统服务(一)
一. 概述 使用java语言开发应用程序,在windows平台下,一般存在3种应用形式: 1. web应用.web应用多数打成war包在web容器(如tomcat,jetty等)中运行. 2. 桌面应 ...
- Struts2 数据驱动
在servlet中获取页面传递过来的数据的方式是:request.getParameter(“username”);这个代码可以获取到页面的username的数据.在action中可以通过属性驱动的方 ...