bzoj 4199 [NOI2015]寿司晚宴
Description
为了庆祝 NOI 的成功开幕,主办方为大家准备了一场寿司晚宴。小 G 和小 W 作为参加 NOI 的选手,也被邀请参加了寿司晚宴。
Input
输入文件的第 1 行包含 2 个正整数 n,p,中间用单个空格隔开,表示共有 n 种寿司,最终和谐的方案数要对 p 取模。
Output
输出一行包含 1 个整数,表示所求的方案模 p 的结果。
Sample Input
Sample Output
HINT
2≤n≤500
Source
这是第二遍码这个题的题解了,第一遍快打完的时候被我菊儿子手贱删掉了
题目要让两个集合中的所有数互质,那么就是相同的质因子不能位于两个不同的集合中
所以这个题的限制和突破口其实就是质因子
我们容易发现选了一个数就是相当与选了这个数的质因子的集合,所以我们需要对每个数进行质因数分解
但是500以内的质数还是比较多的,我们无法通过比较快捷有效的方法每个质因子是否出现过
然后我们就需要知道一个很鬼的东西:
在一个数的所有质因子中最多只有一个会大于sqrt(n);
证明:一个数可以被分解为几个质因子相乘,且这种分解是唯一的,如果有两个质数大于sqrt(n),那么乘积就会大于n,矛盾
那么sqrt(500)==22.36,在这里面的质数只有8个{2,3,5,7,9,11,13,17,19}.
这样一个数小于sqrt(n)的因子就可以用状态压缩来完美判断是否出现,我们还需要对大于sqrt的质因子特殊处理
用一个结构体来存储一个数
一是要记录这个数前8个因子的二进制状态:zt,二是记录大于等于sqrt(n)的因子 greatest是多少,如果没有则记为1
我们把数按照greatest排序那么我们可以把greatest相同的放在一起
我们易发现greatest相同且不为1的必须捆绑处理:必须是只放入第一个人或者只放入第二个人或者都不放入
我们讲到现在就可以设状态了
f[s1][s2]表示全局第一个人选集合s1,第二个人选集合s2的方案数(s1,s2都是表示一个二进制的状态);
g[0][s1][s2],g[1][s1][s2]是临时由f[s1][s2]赋值,再反过来更新f[s1][s2]
g[0][s1][s2]表示第一个人选s1,第二个人选s2,且当前这个数由第一个人选的方案
g[1][s1][s2]则反之
比如把一个数由第一个人选那么我们首先要判断a[i].zt&s2==0,即该数不会与第二个人的集合s2发生矛盾
我们易想到到转移方程
g[0][a[i].zt| s1][s2]+=g[0][s1][s2];
做一遍DP,直到完整的区间结束后由g[0][s1][s2],g[1][s1][s2]赋值给f[s1][s2],这样会保证这个区间的所有数的选择是一样的,不会分居两个集合
但由于都不选的算了两次,所以f[i][j]=g[0][s1][s2]+g[1][s1][s2]-f[s1][s2]
这个题要搞清楚f和g的转化时间,最后统计答案即可
这样就AC了
// MADE BY QT666
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<iostream>
#include<queue>
#include<set>
#include<cstdlib>
#include<cstring>
#include<string>
#include<ctime>
#define lson num<<1
#define rson num<<1|1
#define int long long
using namespace std;
const int MAXS=257;
const int N=550;
const int S=255;
int gi()
{
int x=0,flag=1;
char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-') flag=-1;ch=getchar();}
while(ch>='0'&&ch<='9') x=x*10+ch-'0',ch=getchar();
return x*flag;
}
struct data
{
int zt,greatest;
}a[N];
int n,mod,f[MAXS][MAXS],g[2][MAXS][MAXS];
int biao[10]={2,3,5,7,11,13,17,19};
void pre(){
for(int i=2;i<=n;i++) {
int x=i;
for(int j=0;j<8;j++) {
if(x%biao[j]>0) continue;
a[i].zt|=(1<<j);
while(x%biao[j]==0) x/=biao[j];
}
a[i].greatest=x;
}
}
bool cmp(data a,data b){
return a.greatest<b.greatest;
}
void work(){
for(int i=2;i<=n;i++){
if(i==2||a[i].greatest!=a[i-1].greatest||a[i].greatest==1){
memcpy(g[0],f,sizeof(f));
memcpy(g[1],f,sizeof(f));
}
for(int j=S;j>=0;j--)
for(int k=S;k>=0;k--){
if((j&k)==0){
if((a[i].zt&k)==0)
g[0][a[i].zt|j][k]+=g[0][j][k],g[0][a[i].zt|j][k]%=mod;
if((a[i].zt&j)==0)
g[1][j][a[i].zt|k]+=g[1][j][k],g[1][j][a[i].zt|k]%=mod;
}
}
if(n==2||a[i].greatest!=a[i+1].greatest||a[i].greatest==1){
for(int j=S;j>=0;j--)
for(int k=S;k>=0;k--){
if((j&k)==0){
f[j][k]=g[0][j][k]+g[1][j][k]-f[j][k];
}
}
}
}
}
main()
{
n=gi(),mod=gi();
pre();
sort(a+2,a+1+n,cmp);
f[0][0]=1;
work();
int ans=0;
for(int i=S;i>=0;i--)
for(int j=S;j>=0;j--)
if((i&j)==0)
ans+=f[i][j],ans%=mod;
while(ans<0) ans+=mod;ans%=mod;
printf("%lld",ans);
}
bzoj 4199 [NOI2015]寿司晚宴的更多相关文章
- BZOJ 4197: [Noi2015]寿司晚宴( dp )
N^0.5以内的质数只有8个, dp(i, j, k)表示用了前i个大质数(>N^0.5), 2人选的质数(<=N^0.5)集合分别为j, k时的方案数. 转移时考虑当前的大质数p是给哪个 ...
- BZOJ 4197: [Noi2015]寿司晚宴 状态压缩 + 01背包
4197: [Noi2015]寿司晚宴 Time Limit: 10 Sec Memory Limit: 512 MB Description 为了庆祝 NOI 的成功开幕,主办方为大家准备了一场寿 ...
- [BZOJ]4197: [Noi2015]寿司晚宴
Time Limit: 10 Sec Memory Limit: 512 MB Description 为了庆祝 NOI 的成功开幕,主办方为大家准备了一场寿司晚宴.小 G 和小 W 作为参加 NO ...
- bzoj 4197: [Noi2015]寿司晚宴【状压dp】
一个数内可能多个的质因数只有小于根号n的,500内这样的数只有8个,所以考虑状压 把2~n的数处理出小于根号500的质因数集压成s,以及大质数p(没有就是1),然后按p排序 根据题目要求,拥有一个质因 ...
- BZOJ 4197: [Noi2015]寿司晚宴 状压dp+质因数分解
挺神的一道题 ~ 由于两个人选的数字不能有互质的情况,所以说对于一个质因子来说,如果 1 选了,则 2 不能选任何整除该质因子的数. 然后,我们发现对于 1 ~ 500 的数字来说,只可能有一个大于 ...
- [BZOJ4197][Noi2015]寿司晚宴
4197: [Noi2015]寿司晚宴 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 412 Solved: 279[Submit][Status] ...
- BZOJ_4197_[Noi2015]寿司晚宴_状态压缩动态规划
BZOJ_4197_[Noi2015]寿司晚宴_状态压缩动态规划 Description 为了庆祝 NOI 的成功开幕,主办方为大家准备了一场寿司晚宴.小 G 和小 W 作为参加 NOI 的选手,也被 ...
- [NOI2015]寿司晚宴 --- 状压DP
[NOI2015]寿司晚宴 题目描述 为了庆祝NOI的成功开幕,主办方为大家准备了一场寿司晚宴. 小G和小W作为参加NOI的选手,也被邀请参加了寿司晚宴. 在晚宴上,主办方为大家提供了n−1种不同的寿 ...
- 【BZOJ4197】[Noi2015]寿司晚宴 状压DP+分解质因数
[BZOJ4197][Noi2015]寿司晚宴 Description 为了庆祝 NOI 的成功开幕,主办方为大家准备了一场寿司晚宴.小 G 和小 W 作为参加 NOI 的选手,也被邀请参加了寿司晚宴 ...
随机推荐
- mybatis 一对一关联映射实例
在实际项目开发中,经常存在一对一的关系,如一个人对应一张身份证信息,这就是一对一的关系.下面是一个简单的实例: 1.建表过程我就省略了,主要是一张Person表,一张IDCard表,其相关属性见步骤2 ...
- iOS开发 关于addChildViewController的理解
iOS开发 关于addChildViewController的理解 前言 我之前是做Android开发的接触ios开发不到一个月的时间,所以在有些东理解上会不自觉的向Android方向靠拢. 理解 通 ...
- 微软云计算 Massive Data 处理语言Scope 1
Massive Data处理一直是云计算中很重要的一个环节.目前像Google,Yahoo在相关方面都有自己专有的技术.例如Google的基于MapReduce的Sawzall语言.和Yahoo基于H ...
- Ckeditor与Ckfinder的配合使用,上传图片、水印、修改图片名字为当前日期 asp.net
为了配置出来上传功能,并且还添加水印,修改图片的名字为日期,真的头疼了很久,现在来分享一下自己所做的,也算一点小小的成就吧,顺带帮帮很多还在弄这个的猿们.我是分别用了两种方法.先说低版本的Versio ...
- 实践作业1:测试管理工具实践 Day4
由小组吴辉同学和王俊杰同学负责撰写使用手册,详细记录了环境配置完整过程,以及从软件登陆开始,创建一个测试计划,创建新版本,创建测试用例集,分配测试用例到测试计划,关联到测试用例这一系列完整过程. 刘思 ...
- 负载均衡之nginx
什么是负载均衡负载均衡(Load Balance)是分布式系统架构设计中必须考虑的因素之一,它通常是指,将请求/数据[均匀]分摊到多个操作单元上执行,负载均衡的关键在于[均匀].在使用nginx负载均 ...
- AspNet Core 初步认识
Core 的出现对我我没有很大的影响,当时在Core要发布的时候听到周围的人再聊再谈,我没有去太多关注,就是一个屌丝开发人员. 直到又一次偶然见到一位特别喜欢.net的老开发人员谈起Core时落泪了, ...
- Linux下实现CAD数据的导出
近期公司项目涉及到CAD的导出,而且部署服务器申请不到Windows下的,所以技术上的解决方案就是寻求如何在Linux下实现CAD数据的导出. 于是百度了一下,找了几个相关库和软件. 1.dxflib ...
- POJ 1251 Jungle Roads(最小生成树)
题意 有n个村子 输入n 然后n-1行先输入村子的序号和与该村子相连的村子数t 后面依次输入t组s和tt s为村子序号 tt为与当前村子的距离 求链接全部村子的最短路径 还是裸的最小生成树咯 ...
- Javascript自动化文档工具JSDuck在Windows下的使用心得
作者: zyl910 一.工具比较 为了让前端JavaScript程序更具可维护性,更利于团队开发,文档非常重要.此时便需要使用自动化文档工具了. 我对比了各种JavaScript自动化文档工具,发现 ...