N - 寿司晚宴

HYSBZ - 4197

推荐题解

这个题目我觉得还是很难的,借助题解写出来的,题解还看了很久,现在还是不是很理解。

首先这个数比较大有500,如果直接就像这个题目S - Query on a tree 这样写就超时了,而且也存不下这么大的数。

因为这个500以内的质数太多了,然后看了这么多题解你会发现根号500以内的质数只有8个。

而每一个数大于根号500的质数最多一个,所以我们可以先讨论这个小于根号500的质数,然后再判断这个大于根号500的质数。

讨论完之后再判断这个大于根号500的质数的问题。

然后发现这个题目可以一起计算这两个。

首先把这个按照大于根号500的这个数进行分块,大于根号500的相同的质数分成一块,只有每一个块的结束才可以更新答案。

这个是因为每一块里面我们第一个人选这个或者第二个人选这个,所以如果你每次都去更新答案就不可以保证只有一个人选择这块的这个质数。

其实本来只要一个f数字f[i][j]表示第一个人的状态为 i 第二个人的状态为 j 的方案数,但是因为我们要保证这一块只有一个人选了这个质数,

意思就是第一个人如果要选这个只能向他选了这个的方向转移,第二个人同理。

怎么确定这个转移方向就是加一维 g[k][i][j]  k==0 代表第一个人选了,k==1 代表第二个人选的。

然后之后又不需要分开了,所以又赋值会 f 数组。

差不多就是这样。

总的来说:

1. 既然已知这个是一个状压dp就要去找状压什么,这个和质数有关,所以很明显就是状压质数,根据质因数分解所以可以知道每一个数大于根号x的数最多就一个。

2.状态转移,因为之前写过的那个题目,所以可以知道,首先枚举数,然后枚举状态,这个可以一起枚举这两个人的状态。

3.第二点是不考虑有大于根号500的质数的情况,对于大于根号500 这个需要一个新数组来转移,最后赋值回去。

这个赋值回去之后就表示之后的一个数可以任意选之前的两个状态之间的一个状态,没有冲突。

对于第三点可以这么说,就是在每一块里面我们进行转移,分块的结束我们在重新赋值,之后又分块,但是后面的块可以向前面块中转移。

讲的也不是很清楚,感觉自己还是不是很理解。

#include <cstdio>
#include <cmath>
#include <cstdlib>
#include <cstring>
#include <string>
#include <iostream>
#include <algorithm>
#define inf 0x3f3f3f3f
#define inf64 0x3f3f3f3f3f3f3f3f
using namespace std;
const int maxn = 1e5 + ;
typedef long long ll;
ll dp[ << ][ << ], g[][ << ][ << ];
struct node {
ll s, prime;
}a[maxn];
int p[maxn], isp[maxn], m;
void init() {
for (int i = ; i < ; i++) p[i] = ;
for (int i = ; i*i < ; i++) {
if (p[i]) {
for (int j = i * i; j <= ; j += i) {
p[j] = ;
}
}
}
int m = ;
for (int i = ; i <= maxn; i++) if (p[i]) isp[m++] = i;
}
bool cmp(node a, node b) {
return a.prime < b.prime;
}
int main() {
init();
int n, mod;
scanf("%d%d", &n, &mod);
for (int i = ; i < n; i++) {
int x = i + ;
for (int j = ; j < ; j++) {
if (x%isp[j]) continue;
a[i].s |= ( << j);
while (x%isp[j] == ) x /= isp[j];
}
a[i].prime = x;
}
int ends = ( << );
memset(dp, , sizeof(dp));
dp[][] = ;
sort(a + , a + n, cmp);
for (int i = ; i < n; i++) {
if (i == || a[i].prime != a[i - ].prime || a[i].prime == ) {
memcpy(g[], dp, sizeof(dp));
memcpy(g[], dp, sizeof(dp));
}
for (int j = ends - ; j >= ; j--) {
for (int k = ends - ; k >= ; k--) {
if (j&k) continue;
if ((a[i].s&k) == ) (g[][a[i].s | j][k] += g[][j][k]) %= mod;
if ((a[i].s&j) == ) (g[][j][k | a[i].s] += g[][j][k]) %= mod;
}
}
if (i == n - || a[i].prime != a[i + ].prime || a[i].prime == ) {
for (int j = ends - ; j >= ; j--) {
for (int k = ends - ; k >= ; k--) {
if (j&k) continue;
dp[j][k] = (g[][j][k] + g[][j][k] - dp[j][k] + mod) % mod;
}
}
}
}
ll ans = ;
for (int j = ends - ; j >= ; j--) {
for (int k = ends - ; k >= ; k--) {
if (j&k) continue;
ans += dp[j][k];
ans %= mod;
}
}
ans += mod;
printf("%lld\n", ans%mod);
return ;
}

状压 难

N - 寿司晚宴 HYSBZ - 4197 状压dp的更多相关文章

  1. UOJ129 NOI2015 寿司晚宴 数论、状压DP

    传送门 数论题\(n \leq 500\)肯定是什么暴力算法-- 注意到每一个数\(> \sqrt{n}\)的因子最多只有一个,这意味着\(> \sqrt{n}\)的因子之间是独立的,而只 ...

  2. 【NOI2015】【寿司晚宴】【状压DP】

    Description 为了庆祝 NOI 的成功开幕,主办方为大家准备了一场寿司晚宴. 小 G 和小 W 作为參加 NOI 的选手,也被邀请參加了寿司晚宴. 在晚宴上,主办方为大家提供了 n−1 种不 ...

  3. UOJ #129 / BZOJ 4197 / 洛谷 P2150 - [NOI2015]寿司晚宴 (状压dp+数论+容斥)

    题面传送门 题意: 你有一个集合 \(S={2,3,\dots,n}\) 你要选择两个集合 \(A\) 和 \(B\),满足: \(A \subseteq S\),\(B \subseteq S\), ...

  4. NOI 2015 寿司晚宴 (状压DP+分组背包)

    题目大意:两个人从2~n中随意取几个数(不取也算作一种方案),被一个人取过的数不能被另一个人再取.两个人合法的取法是,其中一个人取的任何数必须与另一个人取的每一个数都互质,求所有合法的方案数 (数据范 ...

  5. 排列perm HYSBZ - 1072(状压dp/暴力)

    Description 给一个数字串s和正整数d, 统计s有多少种不同的排列能被d整除(可以有前导0).例如123434有90种排列能被2整除,其中末位为2的有30种,末位为4的有60种. Input ...

  6. 【bzoj4197】[Noi2015]寿司晚宴 分解质因数+状态压缩dp

    题目描述 为了庆祝 NOI 的成功开幕,主办方为大家准备了一场寿司晚宴.小 G 和小 W 作为参加 NOI 的选手,也被邀请参加了寿司晚宴. 在晚宴上,主办方为大家提供了 n−1 种不同的寿司,编号 ...

  7. 状压DP复习

    深感自己姿势水平之蒻……一直都不是很会状压DP,NOIP又特别喜欢考,就来复习一发…… 题目来源 Orz sqzmz T1 [BZOJ4197][NOI2015]寿司晚宴 (做过)质因数分解最大的质因 ...

  8. 状压DP之LGTB 与序列

    题目 思路 这道题竟然是状压DP,本人以为是数论,看都没看就去打下一题的暴力了,哭 \(A_i\)<=30,所以我们只需要考虑1-58个数,再往后选的话还不如选1更优,注意,1是可以重复选取的, ...

  9. K - Painful Bases 状压dp

    Painful Bases LightOJ - 1021 这个题目一开始看,感觉有点像数位dp,但是因为是最多有16进制,因为限制了每一个数字都不同最多就有16个数. 所以可以用状压dp,看网上题解是 ...

随机推荐

  1. Python 开发工具推荐

    对于开发工具,仁者见仁智者见智,关键是自己喜欢,用着顺手就好,不用刻意去追求别人用的是什么工具. 这里给大家主要推荐三款工具,分别是PyCharm.Sublime Text 3.VS Code,因为这 ...

  2. Extjs更新grid

    基于Extjs4.2 原理是创建一个新的store,来覆盖原有的store. //创建数据 var newdatas = { name: "ly", age: 17, adress ...

  3. 文件密码忘记了怎么办,教你如何使用Python破解密码

    前言 文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理. 作者:轻松学编程小梁 PS:如有需要Python学习资料的小伙伴可以加点击下 ...

  4. 熬夜整理出来的干货:Python+Pycharm+PyQT5可视化程序设计入门

    前言 文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理. 作者:朱淑强 PS:如有需要Python学习资料的小伙伴可以加点击下方链接自 ...

  5. A - Free DIY Tour HDU - 1224

    题目大意:每一个城市都有一定的魅力值,然后有一个有向图,根据这个有向图从1到n+1所获得的魅力的最大值,并输出路径(要求只能从编号娇小的城市到编号较大的城市). 题解:很容易想到最短路+路径纪录.但是 ...

  6. Closest Common Ancestors POJ 1470

    Language: Default Closest Common Ancestors Time Limit: 2000MS   Memory Limit: 10000K Total Submissio ...

  7. 引用传参与reference_wrapper

    本文是<functional>系列的第3篇. 引用传参 我有一个函数: void modify(int& i) { ++i; } 因为参数类型是int&,所以函数能够修改传 ...

  8. Kaggle入门——泰坦尼克号生还者预测

    前言 这个是Kaggle比赛中泰坦尼克号生存率的分析.强烈建议在做这个比赛的时候,再看一遍电源<泰坦尼克号>,可能会给你一些启发,比如妇女儿童先上船等.所以是否获救其实并非随机,而是基于一 ...

  9. 最新VMware虚拟机安装Linux系统-CentOS(详细教程)

    一.前言 最近有网友反应初学Linx不会安装,找了许多教程不是太全面,总会遇到一些要不是启动不了,要不是连不上网,各种问题,为了让大家能够顺利的安装,小乐写了一个非常详细的教程,让大家少入坑. 二.背 ...

  10. 模拟HTTP请求调用controller

    原文参考本人的简书:https://www.jianshu.com/p/0221edbe1598 MockMvc实现了对Http请求的模拟,能够直接使用网络的形式,转换到Controller调用,这样 ...