题目当中有三条限制,我们来逐一考虑。对于第一条限制,每次走动的增加量 \(x_i \le M_x, y_i \le M_y\),可以发现一共走的步数是确定的,那么就相当于解这样两个方程组:

\[x_1 + x_2 + \cdots x_R = Tx
\]
\[y_1 + y_2 + \cdots y_R = Ty
\]

其中 \(x_i \le M_x, y_i \le M_y\),其实是两个独立的方程,最终解的数量实际上是上下两个方程解的数量相乘的结果,于是我们已第一个方程的解为例来思考。可以发现直接计算解的数量是不好算的,但钦定一些位置超过限制其他位置随意的方案是很好算的,于是我们可以令 \(f_i\) 表示钦定有 \(i\) 个位置不合法其他位置随意的方案,令 \(g_i\) 表示恰好有 \(i\) 个位置不合法的方案,那么有:

\[\begin{aligned}
f_i &= \dbinom{R}{i} \times \dbinom{Tx - (Mx + 1) \times i + R - 1}{R - 1}\\
&= \sum\limits_{j = i} ^ R \dbinom{j}{i} g_j
\end{aligned}
\]

根据二项式反演:

\[g_0 = \sum\limits_{i = 0} ^ R (-1) ^ i \dbinom{R}{i} \times \dbinom{Tx - (Mx + 1) \times i + R - 1}{R - 1}
\]

再来考虑第二个条件,不能出现任意一个位置使得 \(x_i = 0, y_i = 0\),同样我们发现还是可以使用二项式定理,令 \(f_i\) 表示钦定有 \(i\) 个位置 \(x_i = 0, y_i = 0\),其他位置随意且满足第一条限制的方案,\(g_i\) 为恰好的方案,那么有:

\[\begin{aligned}
f_i &= \dbinom{R}{i} \sum\limits_{i = 0} ^ {R - i} (-1) ^ i \dbinom{R - i}{i} \times \dbinom{Tx - (Mx + 1) \times i + R - i - 1}{R - i - 1}\\
&= \sum\limits_{j = i} ^ R \dbinom{j}{i} g_j
\end{aligned}
\]

根据二项式反演可得:

\[g_0 = \sum\limits_{i = 0} ^ R (-1) ^ i \dbinom{R}{i} \sum\limits_{i = 0} ^ {R - i} (-1) ^ i \dbinom{R - i}{i} \times \dbinom{Tx - (Mx + 1) \times i + R - i - 1}{R - i - 1}
\]

再来考虑第三条限制,同样可以使用二项式反演,只不过这里钦定的方案可能不是那么好算了。但是我们能发现钦定第三类不合法后这个位置将会被占用,那么我们可以一次考虑每条限制,那么一个 \(dp\) 就可以统计出这些方案,令 \(dp_{i, j, k}\) 表示当前考虑完前 \(i\) 种限制,当前已经钦定了 \(j\) 个位置,当前钦定位置上的增量之和为 \(k\) 的方案,那么同样我们枚举当前新哪增那些位置非法,有(令 \(a_i = k_i\)):

\[dp_{i, j, k} = \sum\limits_{l = 0} ^ {\min\{j, \frac{k}{a_i}\}} \dbinom{R - j + l}{l} dp_{i - 1, j - l, k - l \times a_i}
\]

但是这个 \(dp\) 的复杂度很高,因为第三维涉及到枚举走过的步数之和,但题目当中给了一个条件,所有限制的步数都是 \(G\) 的倍数,那么实际上我们只需要存储 \(k = \frac{k}{G}\) 即可,于是就可以将上面哪个 \(dp\) 的状态改写一下,转移基本没有变。可以发现这个 \(dp\) 第一维枚举的复杂度为 \(K\)(下面称其为 \(n\)),第二维的复杂度为 \(\min\{R, \frac{\min\{T_x, T_y\}}{G}\}\),第三维的枚举上限为 \(\frac{\min\{T_x, T_y\}}{G}\) 转移的复杂度也最多为 \(\frac{\min\{T_x, T_y\}}{G}\) 因此这个 \(dp\) 的复杂度是 \(O(n\min\{R, \frac{\min\{T_x, T_y\}}{G}\}(\frac{\min\{T_x, T_y\}}{G}) ^ 2) \le 50 \times 100 ^ 3 = 5 \times 10 ^ 7\)。

但是需要注意的是,我们最后二项式反演的时候,不能直接做两维的二项式反演,因为我们并不能保证那些恰好选择了 \(j\) 个位置非法的情况下有 \(i\) 个位置的增量之和为 \(k\)。因此,我们需要将所有钦定 \(i\) 个位置非法的方案相加再进行容斥。令 \(F_i\) 为钦定有 \(i\) 个位置非法的方案,那么有:

\[F_i = \sum\limits_{j = i} ^ {\frac{\min\{T_x, T_y\}}{G}} dp_{n, i, j} \times g_{R - i, j}
\]

其中 \(g_{i, j}\) 表示走 \(i\) 步,已经使用了 \(j \times G\) 步的满足前两条限制的方案。再令 \(f_{i, j}\) 表示走 \(i\) 步,已经使用了 \(j \times G\) 步的满足第一条限制的方案,则:

\[g_{i, j} = \sum\limits_{k = 0} ^ i (-1) ^ k \dbinom{i}{k} f_{i - k, j}
\]

可以发现有用的 \(f_{i, j}\) 只有 \(R \times \frac{\min\{T_x, T_y\}}{G}\) 个,每次计算需要 \(R\) 次,因此计算出所有可用的 \(f\) 总共的复杂度为 \(O(R ^ 2 \times \frac{\min\{T_x, T_y\}}{G})\) 还能承受。而可用的 \(g\) 也只有 \(R \times \frac{\min\{T_x, T_y\}}{G}\) 个,预处理出 \(f\) 后复杂度也可以做到 \(O(R ^ 2 \times \frac{\min\{T_x, T_y\}}{G})\)。最后总复杂度 \(O(R ^ 2 \times \frac{\min\{T_x, T_y\}}{G} + n \times (\frac{\min\{T_x, T_y\}}{G}) ^ 3)\) 可以通过本题。

一些坑点

  • 每次二项式反演一定要认真推式子,不然容易出错

  • \(f_0, g_0\) 的情况需要特判

  • 组合数的取值范围到 \(2 \times 10 ^ 6\)

  • 注意枚举 \(dp\) 时第二维和第三维的上限不同

#include<bits/stdc++.h>
using namespace std;
#define rep(i, l, r) for(int i = l; i <= r; ++i)
const int N = 1000000 + 5;
const int M = 100 + 5;
const int K = 1000 + 5;
const int Mod = 1000000000 + 7;
int n, R, G, L1, L2, tx, ty, Tx, Ty, Mx, My, ans;
int a[M], k[M], F[M], fac[N * 2], inv[N * 2], f[K][M], g[K][M], dp[M][M][M];
int read(){
char c; int x = 0, f = 1;
c = getchar();
while(c > '9' || c < '0'){ if(c == '-') f = -1; c = getchar();}
while(c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
return x * f;
}
int Inc(int a, int b){
return (a += b) >= Mod ? a - Mod : a;
}
int Dec(int a, int b){
return (a -= b) < 0 ? a + Mod : a;
}
int Mul(int a, int b){
return 1ll * a * b % Mod;
}
int Qpow(int a, int b){
int ans = 1;
while(b){
if(b & 1) ans = Mul(ans, a);
a = Mul(a, a), b >>= 1;
}
return ans;
}
int C(int n, int m){
if(n < m) return 0;
return Mul(fac[n], Mul(inv[m], inv[n - m]));
}
int main(){
tx = read(), ty = read(), Mx = read(), My = read(), R = read(), G = read();
fac[0] = inv[0] = 1;
rep(i, 1, max(tx, ty) + R) fac[i] = Mul(fac[i - 1], i), inv[i] = Qpow(fac[i], Mod - 2);
n = read();
rep(i, 1, n) a[i] = read();
sort(a + 1, a + n + 1);
n = unique(a + 1, a + n + 1) - a - 1;
L2 = min(tx, ty) / G, L1 = min(R, L2);
if(tx == ty && (tx % G == 0)) f[0][tx / G] = 1;
rep(i, 1, R) rep(j, 0, L2){
int S1 = 0, S2 = 0; Tx = tx - j * G, Ty = ty - j * G; if(Tx < 0 || Ty < 0) continue;
rep(k, 0, i){
if(k & 1) S1 = Dec(S1, Mul(C(i, k), C(Tx - (Mx + 1) * k + i - 1, i - 1)));
else S1 = Inc(S1, Mul(C(i, k), C(Tx - (Mx + 1) * k + i - 1, i - 1)));
if(k & 1) S2 = Dec(S2, Mul(C(i, k), C(Ty - (My + 1) * k + i - 1, i - 1)));
else S2 = Inc(S2, Mul(C(i, k), C(Ty - (My + 1) * k + i - 1, i - 1)));
}
f[i][j] = Mul(S1, S2);
}
if(tx == ty && (tx % G == 0)) g[0][tx / G] = 1;
rep(i, 1, R) rep(j, 0, L2){
rep(k, 0, i){
if(k & 1) g[i][j] = Dec(g[i][j], Mul(C(i, k), f[i - k][j]));
else g[i][j] = Inc(g[i][j], Mul(C(i, k), f[i - k][j]));
}
}
dp[0][0][0] = 1;
rep(i, 1, n) rep(j, 0, L1) rep(k, j, L2) rep(l, 0, min(j, k * G / a[i])){
dp[i][j][k] = Inc(dp[i][j][k], Mul(C(R - j + l, l), dp[i - 1][j - l][k - l * (a[i] / G)]));
}
rep(i, 0, L1) rep(j, i, L2) F[i] = Inc(F[i], Mul(dp[n][i][j], g[R - i][j]));
rep(i, 0, L1) ans = ((i & 1) ? Dec(ans, F[i]) : Inc(ans, F[i]));
printf("%d", ans);
return 0;
}

「SDWC2018 Day1」网格的更多相关文章

  1. LOJ #6374「SDWC2018 Day1」网格

    模拟赛考过的题 当时太菜了现在也一样只拿到了$ 30$分 回来填个坑 LOJ #6374 题意 你要从$ (0,0)$走到$ (T_x,T_y)$,每次移动的坐标增量满足$ 0 \leq \Delta ...

  2. LOJ 6060「2017 山东一轮集训 Day1 / SDWC2018 Day1」Set(线性基,贪心)

    LOJ 6060「2017 山东一轮集训 Day1 / SDWC2018 Day1」Set $ solution: $ 这一题的重点在于优先级问题,我们应该先保证总和最大,然后再保证某一个最小.于是我 ...

  3. LOJ #6060. 「2017 山东一轮集训 Day1 / SDWC2018 Day1」Set

    有趣的思博套路题,想到了基本上加上个对线性基的理解就可以过了 首先考虑到这个把数分成两半的分别异或的过程不会改变某一位上\(1\)的总个数 因此我们求出所有数的\(\operatorname{xor} ...

  4. 【LOJ6060】「2017 山东一轮集训 Day1 / SDWC2018 Day1」Set(线性基)

    点此看题面 大致题意: 让你把\(n\)个数分成两部分,使得在两部分异或和之和最大的前提下,两个异或和中较小的那个尽量小.输出最优的较小异或和. 线性基 关于线性基,可以看一下这篇博客:线性基入门. ...

  5. 题解「2017 山东一轮集训 Day1 / SDWC2018 Day1」Set

    题目传送门 题目大意 给出一个长度为 \(n\) 的数组,选出一些数异或之和为 \(s1\),其余数异或之和为 \(s2\),求 \(s1+s2\) 最大时 \(s1\) 的最小值. 思路 你发现如果 ...

  6. 「雅礼集训 2017 Day1」 解题报告

    「雅礼集训 2017 Day1」市场 挺神仙的一题.涉及区间加.区间除.区间最小值和区间和.虽然标算就是暴力,但是复杂度是有保证的. 我们知道如果线段树上的一个结点,\(max=min\) 或者 \( ...

  7. [LOJ 6031]「雅礼集训 2017 Day1」字符串

    [LOJ 6031] 「雅礼集训 2017 Day1」字符串 题意 给定一个长度为 \(n\) 的字符串 \(s\), \(m\) 对 \((l_i,r_i)\), 回答 \(q\) 个询问. 每个询 ...

  8. [LOJ 6030]「雅礼集训 2017 Day1」矩阵

    [LOJ 6030] 「雅礼集训 2017 Day1」矩阵 题意 给定一个 \(n\times n\) 的 01 矩阵, 每次操作可以将一行转置后赋值给某一列, 问最少几次操作能让矩阵全为 1. 无解 ...

  9. [LOJ 6029]「雅礼集训 2017 Day1」市场

    [LOJ 6029] 「雅礼集训 2017 Day1」市场 题意 给定一个长度为 \(n\) 的数列(从 \(0\) 开始标号), 要求执行 \(q\) 次操作, 每次操作为如下四种操作之一: 1 l ...

随机推荐

  1. Adversarial Examples Are Not Bugs, They Are Features

    目录 概 主要内容 符号说明及部分定义 可用特征 稳定可用特征 可用不稳定特征 标准(standard)训练 稳定(robust)训练 分离出稳定数据 分离出不稳定数据 随机选取 选取依赖于 比较重要 ...

  2. [数据结果]C语言 基础指令汇编

    数据结构 C语言核心操作集合 (1)预定义常量和类型 //函数结果状态代码 #define TRUE 1 #define FALSE 0 #define OK 1 #define ERROR 0 #d ...

  3. CS5211替代PS8625|设计EDP转LVDS转接屏方案|替代PS8625

    PS8625将作为DP或eDP接收器设备出现在视频源中,并将作为LVDS显示面板的LVDS源设备.该设备是一个完全集成的解决方案,不需要外部CPU.内存.时钟基准或电压调节器.PS8625可配置为从显 ...

  4. ECMA-262规范定义的七种错误类型

    第一种:Error    所有错误的基本类型,实际上不会被抛出.   第二种:EvalError   执行eval错误时抛出. 第三种:ReferenceError    对象不存在是抛出. 第四种: ...

  5. Java Record 的一些思考 - 序列化相关

    Java Record 序列化相关 Record 在设计之初,就是为了找寻一种纯表示数据的类型载体.Java 的 class 现在经过不断的迭代做功能加法,用法已经非常复杂,各种语法糖,各种多态构造器 ...

  6. UTF-8,GBK,ANSI之间的关系和区别

    GBK应该是属于ANSI之中的,在ANSI的国际通用集,GBK是专门来解决中文编码的,是双字节的,不论中英文都是双字节,而UTF-8是才用的另外的一种编码方式,对英文是用8位,对中文使用24位,是和A ...

  7. BOM 点击触发 倒计时发送验证码案例

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  8. Python中类的变量,一个下划线与两个下划线的区别

    形似       功能 __xx 这是私有变量, 只有内部可以访问,外部不可以访问.但是也不是一定不可以访问,只要以 _类名__xx样式就可以访问 .但最好不要这样做,养成良好编程习惯 _x 这是实例 ...

  9. 用软碟通UltraISO刻录Win 10 1909 到U盘,只有1个G左右,安装不了系统

    之前一直用软碟通刻录WIN10的ISO镜像到U盘.最近想到用最新版的WIN10 1909 来做一个U盘系统,刻录也成功了.就是安装系统的时候总报错,找了很久原因,终于发现刻录后占用U盘的空间只有1G左 ...

  10. 理解闭包--js面向对象编程

    什么是闭包? 先看一段代码: function a(){ var n = 0; function inc() { n++; console.log(n); } inc(); inc(); } a(); ...