BZOJ.4767.两双手(组合 容斥 DP)
\(Description\)
棋盘上\((0,0)\)处有一个棋子。棋子只有两种走法,分别对应向量\((A_x,A_y),(B_x,B_y)\)。同时棋盘上有\(n\)个障碍点\((x_i,y_i)\),棋子在任何时刻都不能跳到障碍点。
求棋子从\((0,0)\)跳到\((E_x,E_y)\)的方案数。答案对\(10^9+7\)取模。
\(Solution\)
注意到\(A_x*B_y-A_y*B_x\neq0\),即两向量不共线,从某个点走到另一个点,两种方式分别所用次数\(x,y\)是确定的。即求该方程组的非负整数解:$$\left{\begin{array}{lr}A_xx+B_xy=X_i\A_yx+B_yy=Y_i\end{array}\right.$$
同网格图方案数,从某个点以两种方式分别走\(x,y\)步到达另一个点,这样的方案数为\(\binom{x+y}{x}\)。
将每个点表示成这样的\(x,y\)(从\((0,0)\)出发到达该点两种方式分别所需步数)后,任意两点所需的步数就是\(x_i-x_j,y_i-y_j\)了。方案数同样可以用组合数求。
然后就可以排序后容斥了。
\(f(i)\)表示在\(i\)之前不经过任何障碍点,到达障碍点\(i\)的方案数。记\(cnt(i,j)\)表示从障碍点\(i\)到\(j\)的方案数。将起点视为障碍点\(0\),那么$$f(i)=cnt(0,i)-\sum_{j=1}^{i-1}f(j)*cnt(j,i)$$
将终点视为第\(n+1\)个障碍点,答案就是\(f(n+1)\)了。
复杂度\(O(n^2)\)。
因为\(A_x,A_y,B_x,B_y\)可能有负数,所以要走的步数是\(n^2\)级别的(比如\((1,0),(-500,1)\))。组合数要\(n+m\)所以上界要到\(2n^2\)。
//4736kb 448ms 为啥这么慢呢
#include <cstdio>
#include <cctype>
#include <algorithm>
#define gc() getchar()
#define mod 1000000007
typedef long long LL;
const int N=505,M=500000;
int Ax,Ay,Bx,By,f[N],fac[M+2],ifac[M+2];
struct Point
{
int x,y;
bool operator <(const Point &a)const{
return x==a.x?y<a.y:x<a.x;
}
}p[N];
inline int read()
{
int now=0,f=1;register char c=gc();
for(;!isdigit(c);c=='-'&&(f=-1),c=gc());
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now*f;
}
inline int FP(int x,int k)
{
int t=1;
for(; k; k>>=1,x=1ll*x*x%mod)
if(k&1) t=1ll*t*x%mod;
return t;
}
void Calc(int yi,int xi,int &x,int &y)
{
int a=xi*By-yi*Bx,b=Ax*By-Ay*Bx;
if(!b||a%b) {x=-1; return;}//判0。。
int c=xi*Ay-yi*Ax,d=Bx*Ay-By*Ax;
if(!d||c%d) {x=-1; return;}
x=a/b, y=c/d;
}
inline int C(int n,int m)
{
if(n<0||m<0) return 0;//return C(n+m,n)
return 1ll*fac[n+m]*ifac[n]%mod*ifac[m]%mod;
}
int main()
{
fac[0]=fac[1]=1;
for(int i=2; i<=M; ++i) fac[i]=1ll*fac[i-1]*i%mod;
ifac[M]=FP(fac[M],mod-2);
for(int i=M-1; ~i; --i) ifac[i]=1ll*ifac[i+1]*(i+1)%mod;
int Ex=read(),Ey=read(),n=read(),cnt=0;
Ax=read(), Ay=read(), Bx=read(), By=read();
Calc(Ey,Ex,Ex,Ey);
if(Ex<0||Ey<0) return puts("0"),0;
p[++cnt]=(Point){Ex,Ey};
for(int x,y; n--; )
{
Calc(read(),read(),x,y);
if(!(x<0||y<0||x>Ex||y>Ey)) p[++cnt]=(Point){x,y};//需要走更多步的不会相交(否则排序还出问题?)
}
n=cnt, std::sort(p+1,p+1+n);
for(int i=1; i<=n; ++i)
{
Point now=p[i]; LL tmp=C(now.x,now.y);
if(!tmp) continue;
for(int j=1; j<i; ++j)
tmp+=mod-1ll*f[j]*C(now.x-p[j].x,now.y-p[j].y)%mod;
f[i]=(int)(tmp%mod);
}
printf("%d\n",f[n]);
return 0;
}
BZOJ.4767.两双手(组合 容斥 DP)的更多相关文章
- bzoj 4767: 两双手 组合 容斥
题目链接 bzoj4767: 两双手 题解 不共线向量构成一组基底 对于每个点\((X,Y)\)构成的向量拆分 也就是对于方程组 $Ax * x + Bx * y = X $ \(Ay * x + B ...
- bzoj 4767 两双手 - 动态规划 - 容斥原理
题目传送门 传送门I 传送门II 题目大意 一个无限大的棋盘上有一只马,设马在某个时刻的位置为$(x, y)$, 每次移动可以将马移动到$(x + A_x, y + A_y)$或者$(x + B_x, ...
- BZOJ 4767 两双手
题解: 发现这种题目虽然可以想出来,但磕磕碰碰得想挺久的 根据数学可以知道组成方案是唯一的(集合) 然后发现每个使用的大小可能是接近n^2的 直接dp(n^4)是过不了的 那么先观察观察 我们可以把每 ...
- BZOJ 4767: 两双手 [DP 组合数]
传送门 题意: 给你平面上两个向量,走到指定点,一些点不能经过,求方案数 煞笔提一开始被题面带偏了一直郁闷为什么方案不是无限 现在精简的题意.....不就是$bzoj3782$原题嘛,还不需要$Luc ...
- 【BZOJ】4767: 两双手【组合数学】【容斥】【DP】
4767: 两双手 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1057 Solved: 318[Submit][Status][Discuss] ...
- 【BZOJ3622】已经没有什么好害怕的了 容斥+DP
[BZOJ3622]已经没有什么好害怕的了 Description Input Output Sample Input 4 2 5 35 15 45 40 20 10 30 Sample Output ...
- HDU 5794 A Simple Chess (容斥+DP+Lucas)
A Simple Chess 题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5794 Description There is a n×m board ...
- [CF1086E]Beautiful Matrix(容斥+DP+树状数组)
给一个n*n的矩阵,保证:(1)每行都是一个排列 (2)每行每个位置和上一行对应位置不同.求这个矩阵在所有合法矩阵中字典序排第几.考虑类似数位DP的做法,枚举第几行开始不卡限制,那么显然之前的行都和题 ...
- $bzoj2560$ 串珠子 容斥+$dp$
正解:容斥+$dp$ 解题报告: 传送门$QwQ$ $umm$虽然题目蛮简练的了但还是有点难理解,,,我再抽象一点儿,就说有$n$个点,点$i$和点$j$之间有$a_{i,j}$条无向边可以连,问有多 ...
随机推荐
- C语言练手游戏-控制台输出一个会移动的坦克
把C语言的知识融合起来做一个练手的小游戏项目,将自己掌握到的数据结构.数组.函数.宏定义等知识综合利用,增加对语法的熟练程度. 操作系统: windows 10 x64 编译IDE : VS2015 ...
- html5 postMessage解决跨域、跨窗口消息传递(转)
仅做学习使用,原文链接:http://www.cnblogs.com/dolphinX/p/3464056.html 一些麻烦事儿 平时做web开发的时候关于消息传递,除了客户端与服务器传值还有几个经 ...
- table下tbody滚动条与thead对齐的方法且每一列可以不均等
1 前言 table下tbody滚动条与thead对齐的方法,开始在tbody的td和thead的tr>td,对每一个Item加入百分比,结果是没对齐.也尝试了用bootstrap的col-md ...
- ajax post 传递数组参数
1.前言 此文章仅作为记录,方便查阅. 2.代码 javascript: var idArr = ['one','two','Three']; $.ajax({ type: 'POST', data ...
- 一台电脑,两个及多个git账号配置
1. 生成两[三]个ssh公钥私钥 方法参照:http://www.cnblogs.com/fanbi/p/7772812.html第三步骤 假定其中一个是id_rsa, 另一个时id_rsa_two ...
- vue系列之webstrom开发vue,无法热更新
用vue-cli构建了项目之后在webstorm开发,用npm run dev跑本地服务,经常修改之后在浏览器刷新没反应,偶尔才会有刷新,需要重新跑一遍npm run dev才会更新,这是怎么回事呢? ...
- LeetCode(46):全排列
Medium! 题目描述: 给定一个没有重复数字的序列,返回其所有可能的全排列. 示例: 输入: [1,2,3] 输出: [ [1,2,3], [1,3,2], [2,1,3], [2,3,1], [ ...
- LeetCode(29): 两数相除
Medium! 题目描述: 给定两个整数,被除数 dividend 和除数 divisor.将两数相除,要求不使用乘法.除法和 mod 运算符. 返回被除数 dividend 除以除数 divisor ...
- MFC单文档
一.创建并运行MFC单文档程序 1.创建单文档程序 这里使用的是VS2017.首先,打开VS2017,选择文件->新建->项目,然后选择Visual C++ -> MFC /ATL& ...
- LeetCode | Reverse Words in a String(C#)
题目: Given an input string, reverse the string word by word. For example,Given s = "the sky is b ...