uva11916 bsgs算法逆元模板,求逆元,组合计数
其实思维难度不是很大,但是各种处理很麻烦,公式推导到最后就是一个bsgs算法解方程
/*
要给M行N列的网格染色,其中有B个不用染色,其他每个格子涂一种颜色,同一列上下两个格子不能染相同的颜色
涂色方案%100000007的结果是R,现在给出R,N,K,请求出最小的M
对于第一行来说,每个位置有k种选择,那么填色方案数是k^n
对于第二行来说,每个位置有k-1中选择,那么填色方案数时(k-1)^n种
依次类推,如果i+1行的某个格子上面是白格,那么这个格子有k种填色方案 将M行分为两部分,第一部分是固定的,即行数最大的B向下一行,注意特判情况
第二部分是不固定的,即不停增加行数M,直到求出结果=R 另P=(K-1)^N,所以方案总数是cnt*P^M=R (mod 100000007)
P^M = cnt^-1 * R(mod 100000007)
逆元算一下即可
用bsgs算法 解出这个关于M的方程即可
*/
#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define maxn 510
#define mod 100000007 int n,m,k,b,r,x[maxn],y[maxn];
set<pair<int,int> >best; ll pow_mod(ll a,ll p){//快速幂
ll res=;
while(p){
if(p%)
res=res*a%mod;
p>>=;
a=a*a%mod;
}
return res;
}
ll exgcd(ll a,ll b,ll &x,ll &y){
if(b==){x=;y=;return a;}
ll d=exgcd(b,a%b,y,x);
y-=a/b*x;
return d;
}
ll inv(ll a){//ax+y*mod=1 ==> ax=1(mod mod),所以x就是a关于mod的逆元
ll d,x,y;
d=exgcd(a,mod,x,y);
return d==?(x+mod)%mod:-;
}
int log_mod(int a,int b){//bsgs算法,求解a^x=b(mod m)方程
int m, v, e = , i;
m = (int)sqrt(mod+0.5);
v = inv(pow_mod(a, m));
map<int, int> x;
x[] = ; for(int j=;j<m;j++){//建立hash表,x=i*m+j
e=(ll)e*a%mod;
if(!x.count(e))
x[e]=j;
}
for(int i=;i<m;i++){
if(x.count(b))
return i*m+x[b];
b=(ll)b*v%mod;//这里实际上是用逆元处理了,即将a^(i*m+j)=b (mod m)转化为a^j=b^(i*m)^(-1) (mod m)
}
return -;
}
int count(){//计算固定部分的方案数
int c=;//统计b块下面的的方格
for(int i=;i<b;i++)
if(x[i]!=m && !best.count(make_pair(x[i]+,y[i])))
c++;
c+=n;
for(int i=;i<b;i++)
if(x[i]==)
c--; return pow_mod(k-,(ll)m*n-b-c)*pow_mod(k,c)%mod;
}
int doit(){
int cnt=count();//先求出第一部分的cnt
if(cnt==r)
return m; int c=;//要把第m+1行单独拿出来考虑
for(int i=;i<b;i++)
if(x[i]==m)
c++;
m++;
cnt=cnt*pow_mod(k,c)%mod;
cnt=cnt*pow_mod(k-,n-c)%mod;
if(cnt==r)
return m; //接下去就只要求对数方程即可
int P=pow_mod(k-,n);
return log_mod(P,r*inv(cnt)%mod)+m;
}
int main(){
int t,cas=;
scanf("%d",&t);
while(t--){
scanf("%d%d%d%d",&n,&k,&b,&r);
best.clear();
m=;
for(int i=;i<b;i++){
scanf("%d%d",&x[i],&y[i]);
m=max(x[i],m);
best.insert(make_pair(x[i],y[i]));
}
printf("Case %d: %d\n",cas++,doit());
}
}
uva11916 bsgs算法逆元模板,求逆元,组合计数的更多相关文章
- 公钥密码之RSA密码算法扩展欧几里德求逆元!!
扩展欧几里得求逆元 实话说这个算法如果手推的话问题不大,无非就是辗转相除法的逆过程,还有一种就是利用扩展欧几里德算法,学信安数学基础的时候问题不大,但现在几乎都忘了,刷题的时候也是用kuangbin博 ...
- BSGS算法(模板)
BSGS (大步小步算法) 已知\(a.b. c\),求\(x\).令\(a^x \equiv b \pmod c\). 步骤 \[m = \lceil \sqrtc\ \rceil \]\[x = ...
- 算法竞赛进阶指南0x36组合计数
概述 AcWing211. 计算系数 #include <bits/stdc++.h> using namespace std; const int mod = 10007 ; int k ...
- hdu 1576 求逆元
题意:给出n=A mod 9973和B,求(A/B) mod 9973 昨天用扩展欧几里得做过这题,其实用逆元也可以做. 逆元的定义:例如a*b≡1 (mod m),则b就是a关于m的逆元. 求逆元方 ...
- POJ2417 Discrete Logging | A,C互质的bsgs算法
题目: 给出A,B,C 求最小的x使得Ax=B (mod C) 题解: bsgs算法的模板题 bsgs 全称:Baby-step giant-step 把这种问题的规模降低到了sqrt(n)级别 首 ...
- Codeforces 451E Devu and Flowers(组合计数)
题目地址 在WFU(不是大学简称)第二次比赛中做到了这道题.高中阶段参加过数竞的同学手算这样的题简直不能更轻松,只是套一个容斥原理公式就可以.而其实这个过程放到编程语言中来实现也没有那么的复杂,不过为 ...
- 多项式求逆元详解+模板 【洛谷P4238】多项式求逆
概述 多项式求逆元是一个非常重要的知识点,许多多项式操作都需要用到该算法,包括多项式取模,除法,开跟,求ln,求exp,快速幂.用快速傅里叶变换和倍增法可以在$O(n log n)$的时间复杂度下求出 ...
- BSGS算法+逆元 POJ 2417 Discrete Logging
POJ 2417 Discrete Logging Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 4860 Accept ...
- 线性筛prime/phi/miu/求逆元模板
这绿题贼水...... 原理我不讲了,随便拿张草稿纸推一下就明白了. #include <cstdio> using namespace std; ; int su[N],ans,top; ...
随机推荐
- 简单重写容器vector
#pragma once #include <iostream> using namespace std; template<class T> class CMyVector ...
- Python 爬虫六 性能相关
前面已经讲过了爬虫的两大基础模块: requests模块:用来伪造请求爬取数据 bs4模块:用来整理,提取数据 当我们真正的开始有需求的时候通常都是批量爬取url这样的.那如何批量爬取呢? 按照正常的 ...
- Coconuts HDU - 5925 (二维离散化求连通块的个数以及大小)
题目链接: D - Coconuts HDU - 5925 题目大意:首先是T组测试样例,然后给你n*m的矩阵,原先矩阵里面都是白色的点,然后再输入k个黑色的点.这k个黑色的点可能会使得原先白色的点 ...
- 人人项目renren-security\git\renren-security的目录下的文件列表
\.git\config; \.git\FETCH_HEAD; \.git\HEAD; \.git\index; \.git\logs\HEAD; \.git\logs\refs\heads\mast ...
- 调试 - Visual Studio调试
Visual Studio - 调试 异常处理机制 windows预定义了一系列的异常错误码,每种程序异常都有一个对应的错误码,windows系统将这些类似键值对关系的数据存储在异常处理表中(称为SE ...
- js中创建数组,并往数组里添加元素
数组的创建 var arrayObj = new Array(); //创建一个数组 var arrayObj = new Array([size]); //创建一个数组并指定长度,注意不是上限,是长 ...
- python,获取用户输入,并且将输入的内容写到.txt
该功能缺点是必须保证该文件不存在的情况才会成功 f=open('E:/mywork/保存文件.txt','x') def userwrite(code): if code=='w': f.close( ...
- Boost 序列化
原文链接: https://blog.csdn.net/qq2399431200/article/details/45621921 1. 编译器 gcc, boost 1.55 2.1第一个简单的例子 ...
- k64 datasheet学习笔记45---10/100-Mbps Ethernet MAC(ENET)之概述
1.前言 k64 ENET CORE 实现了10M/100Mbps的Ethernet MAC,与IEEE802.3-2002标准兼容. MAC层与全双工/半双工的10M/100Mbps以太网兼容: M ...
- longzhuapp项目笔记
1.配置不同环境的打包命令