SSY的队列 hash+记忆化
题目描述
\(SSY\) 是班集体育委员,总喜欢把班级同学排成各种奇怪的队形,现在班级里有 \(N\) 个身高互不相同的同学,请你求出这 \(N\) 个人的所有排列中任意两个相邻同学的身高差均不为给定整数M的倍数的排列总数。
输入格式
共三行:
第一行为 \(N\)
第二行为 \(N\) 个不同的整数
第三行为 \(M\)
输出格式
一行,为符合条件的排列总数(答案对 \(1234567891\) 取余数)。
样例
样例输入1
3
-1 0 1
2
样例输出1
2
样例输入2
4
1 2 3 4
3
样例输出2
12
数据范围与提示
\(20\%\) 的数据:\(N<=11\)
\(70\%\) 的数据:\(N<=15\)
\(100\%\) 的数据:\(N<=30,M<=1000\)。
分析
对于一个数,它原来的值和它对 \(m\) 取模之后的值在这道题中意义是相同的
所以一共只会有 \(m\) 种数
我们记录一下每一种数有多少个,然后把个数存进栈里
我们会发现,方案数仅与当前的数和剩下个数为 \(x\) 的数有几种有关
比如当 \(m=5\) 时,\(3\ 2\ 2\)和 \(3\ 1\ 1\) 的结果是完全一样的
可以设 \(f[i][j][k][...]\) 为当前选的是第 \(i\) 种数,剩下个数为 \(1\) 的数有 \(j\) 种,剩下个数为 \(2\) 的数有 \(k\) 种 ... 的方案数
数组开不下,所以我们可以用哈希的思想把状态压成一个
对于递归的每一层,开一个 \(map\) 记录一下即可
要注意的是最后要乘上每一种数个数的阶乘,因为同一种数可以任意交换位置
代码
#include<cstdio>
#include<map>
#include<algorithm>
#include<cstring>
#include<cmath>
#define rg register
inline int read(){
rg int x=0,fh=1;
rg char ch=getchar();
while(ch<'0' || ch>'9'){
if(ch=='-') fh=-1;
ch=getchar();
}
while(ch>='0' && ch<='9'){
x=(x<<1)+(x<<3)+(ch^48);
ch=getchar();
}\
return x*fh;
}
#define ull unsigned long long
const int maxn=35;
const int mod=1234567891;
const ull bas=233333;
int a[maxn],sta[maxn],b[maxn],n,m,tp,mmax,jc[maxn];
bool vis[maxn];
std::map<ull,int> mp[maxn];
int dfs(int now,int lat){
if(now>n){
return 1;
}
memset(b,0,sizeof(b));
for(rg int i=1;i<=tp;i++){
if(i!=lat) b[sta[i]]++;
}
ull nans=sta[0];
for(rg int i=0;i<=mmax;i++){
nans=nans*bas+b[i];
}
nans=nans*bas+sta[lat];
if(mp[now].find(nans)!=mp[now].end()) return mp[now][nans];
rg int mans=0;
if(sta[0]>0){
sta[0]--;
mans=((long long)mans+(long long)dfs(now+1,0))%mod;
sta[0]++;
}
for(rg int i=1;i<=tp;i++){
if(i!=lat && sta[i]>0){
sta[i]--;
mans=((long long)mans+(long long)dfs(now+1,i))%mod;
sta[i]++;
}
}
mp[now][nans]=mans;
return mans;
}
int main(){
n=read();
for(rg int i=1;i<=n;i++){
a[i]=read();
}
m=read();
for(rg int i=1;i<=n;i++){
a[i]%=m;
if(a[i]<0) a[i]+=m;
}
rg int ncnt=0;
for(rg int i=1;i<=n;i++){
if(vis[i]) continue;
vis[i]=1;
ncnt=0;
for(rg int j=i;j<=n;j++){
if(a[i]==a[j]){
vis[j]=1;
ncnt++;
}
}
mmax=std::max(mmax,ncnt);
if(ncnt==1) sta[0]++;
else sta[++tp]=ncnt;
}
jc[0]=1;
for(rg int i=1;i<=n;i++){
jc[i]=1LL*jc[i-1]*i%mod;
}
rg int ans=1;
for(rg int i=0;i<=tp;i++){
ans=1LL*ans*jc[sta[i]]%mod;
}
ans=1LL*ans*dfs(1,0)%mod;
printf("%d\n",ans);
return 0;
}
SSY的队列 hash+记忆化的更多相关文章
- [HNOI2013]比赛 (用Hash实现记忆化搜索)
[HNOI2013]比赛 题目描述 沫沫非常喜欢看足球赛,但因为沉迷于射箭游戏,错过了最近的一次足球联赛.此次联 赛共N支球队参加,比赛规则如下: (1) 每两支球队之间踢一场比赛. (2) 若平局, ...
- TC-572-D1L2 (双向搜索+记忆化)
solution: 这一题是比较难实现的双向搜索题:(字符串+双向搜索+hash记忆化) 我们可以先把K的前半部分枚举出来,并将得出的所有结果和题目给的n个数的每一个数的前半部分都比对一遍,得到它和每 ...
- LuoguP2254 [NOI2005]瑰丽华尔兹 (单调队列优化DP)(用记忆化过了。。。)
记忆化 #include <cstdio> #include <iostream> #include <cstring> #include <algorith ...
- FZU 2092 bfs+记忆化搜索
晚上团队训练赛的题 和普通bfs不同的是 这是同时操纵人与影子两个单位进行的bfs 由于可能发生人和影子同时接触水晶 所以不可以分开操作 当时使用node记录人和影子的位置 然后进行两重for循环来分 ...
- FZU 2092 收集水晶 bfs+记忆化搜索 or 暴力
题目链接:收集水晶 一眼看过去,觉得是普通的bfs,初始位置有两个.仔细想了想...好像如果这样的话..........[不知道怎么说...T_T] dp[12][12][12][12][210] 中 ...
- (中等) POJ 1054 The Troublesome Frog,记忆化搜索。
Description In Korea, the naughtiness of the cheonggaeguri, a small frog, is legendary. This is a we ...
- (区间dp + 记忆化搜索)Treats for the Cows (POJ 3186)
http://poj.org/problem?id=3186 Description FJ has purchased N (1 <= N <= 2000) yummy treats ...
- hdu1428 记忆化搜索(BFS预处理最短路径和+DP+DFS)
题意:有一块 n * n 大小的方形区域,要从左上角 (1,1)走到右下角(n,n),每个格子都有通过所需的时间,并且每次所走的下一格到终点的最短时间必须比当前格子走到重点的最短时间短,问一共有多少种 ...
- tyvj 1004 滑雪 记忆化搜索
滑雪 Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.tyvj.cn/p/1004 Description trs喜欢滑雪.他来 ...
随机推荐
- 苏州6617.9373(薇)xiaojie:苏州哪里有xiaomei
苏州哪里有小姐服务大保健[微信:6617.9373倩儿小妹[苏州叫小姐服务√o服务微信:6617.9373倩儿小妹[苏州叫小姐服务][十微信:6617.9373倩儿小妹][苏州叫小姐包夜服务][十微信 ...
- pytest文档53-命令行实时输出错误信息(pytest-instafail)
前言 pytest 运行全部用例的时候,在控制台会先显示用例的运行结果(.或F), 用例全部运行完成后最后把报错信息全部一起抛出到控制台. 这样我们每次都需要等用例运行结束,才知道为什么报错,不方便实 ...
- pytest文档48-切换 base_url 测试环境(pytest-base-url)
前言 当我们自动化代码写完成之后,期望能在不同的环境测试,这时候应该把 base_url 单独拿出来,能通过配置文件和支持命令行参数执行. pytest-base-url 是 pytest 里面提供的 ...
- elk-架构图
- python并发编程之多进程(实践篇) 转
一 multiprocessing模块介绍 python中的多线程无法利用多核优势,如果想要充分地使用多核CPU的资源,在python中大部分情况需要使用多进程.Python提供了multiproce ...
- centos8平台使用ulimit做系统资源限制
一,ulimit的用途 1, ulimit 用于shell启动进程所占用的资源,可用于修改系统资源限制 2, 使用ulimit -a 可以查看当前系统的所有限制值 使用ulimit -n <可以 ...
- Redis Lua脚本完全入门
1. 前言 Redis是高性能的KV内存数据库,除了做缓存中间件的基本作用外还有很多用途,比如胖哥以前分享的Redis GEO地理位置信息计算.Redis提供了丰富的命令来供我们使用以实现一些计算.R ...
- 链接WPA2-企业WIFI时出现无法链接到该网络,可以链接个人WIFI时的问题和解决方案
因在一个问题上掉两次坑所以还是决定记录下来,方便以后查阅. 第一次因为要部署.net 应用程序要求使用TLS1.2,所以修改了操作系统的默认启用的安全协议类型,导致好多应用程序出问题. 第二次因为vs ...
- vant学习网址
https://youzan.github.io/vant/#/zh-CN/swipe-cell 我推荐大家可以在这个网址上面学习vant相关的知识 里面有代码解说和详细的代码
- Paillier同态加密的介绍以及c++实现
我们先来简短认识一下Paillier同态加密算法: 如果就这么按照定义来用最简朴的c++程序写 就像这样: #include <iostream> #include <math.h& ...