题目描述

\(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+记忆化的更多相关文章

  1. [HNOI2013]比赛 (用Hash实现记忆化搜索)

    [HNOI2013]比赛 题目描述 沫沫非常喜欢看足球赛,但因为沉迷于射箭游戏,错过了最近的一次足球联赛.此次联 赛共N支球队参加,比赛规则如下: (1) 每两支球队之间踢一场比赛. (2) 若平局, ...

  2. TC-572-D1L2 (双向搜索+记忆化)

    solution: 这一题是比较难实现的双向搜索题:(字符串+双向搜索+hash记忆化) 我们可以先把K的前半部分枚举出来,并将得出的所有结果和题目给的n个数的每一个数的前半部分都比对一遍,得到它和每 ...

  3. LuoguP2254 [NOI2005]瑰丽华尔兹 (单调队列优化DP)(用记忆化过了。。。)

    记忆化 #include <cstdio> #include <iostream> #include <cstring> #include <algorith ...

  4. FZU 2092 bfs+记忆化搜索

    晚上团队训练赛的题 和普通bfs不同的是 这是同时操纵人与影子两个单位进行的bfs 由于可能发生人和影子同时接触水晶 所以不可以分开操作 当时使用node记录人和影子的位置 然后进行两重for循环来分 ...

  5. FZU 2092 收集水晶 bfs+记忆化搜索 or 暴力

    题目链接:收集水晶 一眼看过去,觉得是普通的bfs,初始位置有两个.仔细想了想...好像如果这样的话..........[不知道怎么说...T_T] dp[12][12][12][12][210] 中 ...

  6. (中等) POJ 1054 The Troublesome Frog,记忆化搜索。

    Description In Korea, the naughtiness of the cheonggaeguri, a small frog, is legendary. This is a we ...

  7. (区间dp + 记忆化搜索)Treats for the Cows (POJ 3186)

    http://poj.org/problem?id=3186   Description FJ has purchased N (1 <= N <= 2000) yummy treats ...

  8. hdu1428 记忆化搜索(BFS预处理最短路径和+DP+DFS)

    题意:有一块 n * n 大小的方形区域,要从左上角 (1,1)走到右下角(n,n),每个格子都有通过所需的时间,并且每次所走的下一格到终点的最短时间必须比当前格子走到重点的最短时间短,问一共有多少种 ...

  9. tyvj 1004 滑雪 记忆化搜索

    滑雪 Time Limit: 1 Sec  Memory Limit: 256 MB 题目连接 http://www.tyvj.cn/p/1004 Description     trs喜欢滑雪.他来 ...

随机推荐

  1. 在shell中截取心仪的字符串

    file=/dir1/dir2/dir3/my.file.txt ${file#*/} 去掉左边 ${file##*/} 去掉左边最后一个 ${file%/*} 去掉右边 ${file%%/*} 去掉 ...

  2. 基于python实现单链表代码

    1 """ 2 linklist.py 3 单链表的构建与功能操作 4 重点代码 5 """ 6 7 class Node: 8 " ...

  3. 落地Azure CosmosDb的一个项目分享

    我们遇到了什么? 我们有这么一个业务场景,就是某供应商会去爬取某些数据,爬到后会发到一个FTP上,然后我们定时去获取这些数据 这个数据有大有小,小的30多M数据量百万级,大的数据量能到数百M上千万数据 ...

  4. rabbitmq 延时队列

    前言 某个产品 或者订单,有个有效期 过了有效期要取消 方法一 : 写个脚本,用crontab 定时扫描 改变状态 但是最低只能一分钟 ,不适合 方法二 : 用swoole得毫秒定时器,每秒钟去扫描表 ...

  5. mac安装go环境

    下载pkg文件 https://golang.google.cn/dl/   sudo vim /etc/profile export GOROOT=/usr/local/go export GOPA ...

  6. ASP.NET实现进度条效果【转】

     原文地址:http://www.jb51.net/article/115310.htm 这篇文章主要为大家详细介绍了ASP.NET实现简单的进度条效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一 ...

  7. Java8中Stream 的一些用法

    数据结构和数据准备 @Data @AllArgsConstructor @NoArgsConstructor static class StreamItem { Integer id; String ...

  8. 使用PyPdf2合并PDF文件(没有空白、报错)

    使用PyPdf2合并PDF文件(没有空白.报错) 对于合并之后pdf空白,或者出现 'latin-1' codec can't encode characters in position 8-11: ...

  9. linux ssh自动输入密码,expect使用

    想搞一个使用ssh登录批量ip地址执行命令,自动输入密码的脚本,但是ssh不能使用标准输入来实现自动输入密码,于是了解到了expect这个可以交互的命令 是什么 查看使用man查看expect,是这么 ...

  10. JUC---11单例模式

    一.什么是单例模式 单例模式(Singleton Pattern)是 Java 中最简单的设计模式之一.这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式.这种模式涉及到一个单一的类,该 ...