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喜欢滑雪.他来 ...
随机推荐
- selenium环境配置学习笔记
一 为什么进行自动化测试 缩短测试周期 避免人为出错 测试信息存储 轻易获取覆盖率 二 web/ui自动化条件和适用范围 手工测试已经完成,后期在不影响进度的前提下逐渐实现自动化 项目周期长,重复性工 ...
- 扫描仪扫描文件处理-Python批量处理
多进程处理扫描出来的图片,参见: https://github.com/barrer/scan-helper bug问题反馈github提Issues
- 【状态压缩DP】HDU 4352 XHXJ'S LIS
题目大意 Vjudge链接 定义一个数的内部LIS长度表示这个数每个数位构成的序列的LIS长度,给出区间\([l,r]\),求区间内内部LIS长度为\(k\)的数的个数. 输入格式 第一行给出数据组数 ...
- Linux系统快速搭建LAMP
LAMP是 Linux + Apache + MySQL + PHP 的环境要求,即web服务器. 1.前置条件: (1)搭建Linux系统(参考博客:https://www.cnblogs.com/ ...
- Redis6 安装
在centos7.5服务器上按照官方发布的安装方式并不能进行正确的安装,现收集并整理如下安装方式,亲测有效 1.安装依赖 yum install -y cpp binutils glibc glibc ...
- Gitlab 11.9.1 高可用教程
Gitlab 11.9.1 高可用教程 一. PostgreSQL数据迁移 由于默认Gitlab的安装会内置Postgres数据库,并且没有对外,所以我们需要通过设置对应的Gitlab的配置将其中的数 ...
- JUC之线程池-三大方法-七大参数-四种拒绝策略
线程池:重点 三大方法 七大参数 四种拒绝策略 使用池化技术的理由: 我们的程序伴随着创建销毁线程十分浪费资源, 所以使用线程池,先创建线程,随用随取,用完归还 简单来说就是节约了资源. 使用线程池的 ...
- js 值类型与引用类型
说明之前先提一个提问题,看一下你是怎么理解的 1. 值类型 简单的数据类型,存放在栈中 var num = 100; var num2 = num; num += 100; console.log(n ...
- sql布尔盲注和时间盲注的二分脚本
布尔盲注: import requests url = "http://challenge-f0b629835417963e.sandbox.ctfhub.com:10080/" ...
- Linux exec族函数解析
背景 在提到 vfork 函数时,我们提到了这个概念.为了更好地学习与运用,我们对exec族函数进行展开. exec函数族 介绍 有时我们希望子进程去执行另外的程序,exec函数族就提供了一个在进程中 ...