bzoj 2734 集合选数
Written with StackEdit.
Description
《集合论与图论》这门课程有一道作业题,要求同学们求出\(\{1, 2, 3, 4, 5\}\)的所有满足以 下条件的子集:若 \(x\) 在该子集中,则 \(2x\) 和 \(3x\) 不能在该子集中。同学们不喜欢这种具有枚举性 质的题目,于是把它变成了以下问题:对于任意一个正整数 \(n\leq 100000\),如何求出\(\{1, 2,..., n\}\) 的满足上述约束条件的子集的个数(只需输出对 \(1000000001\) 取模的结果),现在这个问题就 交给你了。
Input
只有一行,其中有一个正整数 \(n\),\(30\%\)的数据满足 \(n\leq20\)。
Output
仅包含一个正整数,表示\(\{1, 2,..., n\}\)有多少个满足上述约束条件 的子集。
Sample Input
4
Sample Output
8
【样例解释】
有\(8\) 个集合满足要求,分别是\(\emptyset\),\(\{1\}\),\(\{1,4\}\),\(\{2\}\),\(\{2,3\}\),\(\{3\}\),\(\{3,4\}\),\(\{4\}\).
Solution
- 神仙构造题.
- 考虑构造这样的一个矩阵
1 & 3 & 9 & 27 & ...\\
2 & 6 & 18 & 54 & ...\\
4 & 12 & 36 & 108 & ...\\
8 & 24 & 72 & 216 & ...\\
... & ... & ... & ... & ...\\
\end{pmatrix} \quad\]
- 其中一个数是它上边那个数的\(2\)倍,左边那个数的\(3\)倍.
- 原问题转化为从矩阵中选出一些互不相邻的数.
- 那么这个矩阵中的数是呈指数级增长的,规模不会超过\(20\).
- 使用经典状压\(dp\)处理,逐行考虑.
- 另,对于每个尚未出现过的数,需以它为左上角建一个矩阵,这样各个构造出的矩阵互不相交,利用乘法原理统计答案.
左上角的数增大时,矩阵没有填入数字的部分也不断增大.手动\(memset\).
#include<bits/stdc++.h>
using namespace std;
typedef long long LoveLive;
inline int read()
{
int out=0,fh=1;
char jp=getchar();
while ((jp>'9'||jp<'0')&&jp!='-')
jp=getchar();
if (jp=='-')
{
fh=-1;
jp=getchar();
}
while (jp>='0'&&jp<='9')
{
out=out*10+jp-'0';
jp=getchar();
}
return out*fh;
}
const int P=1e9+1;
inline int add(int a,int b)
{
return (a + b) % P;
}
inline int mul(int a,int b)
{
return 1LL * a * b % P;
}
const int MAXN=1e5+10;
int n;
int vis[MAXN];
int Martix[20][20];
int limit[20];//第i行的边界
int ans=1;
int r,c;
int judge(int st)
{
int ls=0;
while(st)
{
int p=st&1;
if(p && ls)
return 0;
ls=p;
st>>=1;
}
return 1;
}
vector<int> G;
void Build_Martix(int x)
{
memset(Martix,-1,sizeof Martix);
r=0,c=0;
int p;
for(int i=1;i<20;++i)
{
p=i==1?x:Martix[i-1][1]*2;
if(p>n)
{
r=i-1;
break;
}
Martix[i][1]=p;
vis[p]=1;
for(int j=2;j<20;++j)
{
p=Martix[i][j-1]*3;
if(p>n)
{
c=max(c,j-1);
limit[i]=j-1;
break;
}
Martix[i][j]=p;
vis[p]=1;
}
}
G.clear();
int S=1<<c;
for(int i=0;i<S;++i)
if(judge(i))
G.push_back(i);
}
int f[20][1<<20];
inline int check(int x,int y)
{
return !(x&y);
}
int dfs(int k,int st)//填好了前k行,且第k行状态为st的方案数.
{
if(k==1)
return 1;
if(f[k][st]!=-1)
return f[k][st];
int &res=f[k][st];
res=0;
int S=(1<<limit[k-1])-1;
for(int v=0;v<G.size();++v)
{
int i=G[v];
if(i>S)
break;
if(check(i,st))
res=add(res,dfs(k-1,i));
}
return res;
}
void solve(int x)//以x为左上角构造矩阵的方案数
{
Build_Martix(x);
for(int i=1;i<=r+1;++i)
{
int S=1<<limit[i];
for(int j=0;j<S;++j)
f[i][j]=-1;
}
//memset(f,-1,sizeof f);
ans=mul(ans,dfs(r+1,0));
}
int main()
{
n=read();
for(int i=1;i<=n;++i)
if(!vis[i])
solve(i);
printf("%d\n",ans);
return 0;
}
bzoj 2734 集合选数的更多相关文章
- bzoj 2734: [HNOI2012]集合选数 状压DP
2734: [HNOI2012]集合选数 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 560 Solved: 321[Submit][Status ...
- 2734: [HNOI2012]集合选数
2734: [HNOI2012]集合选数 链接 分析: 转化一下题意. 1 3 9 27... 2 6 18 54... 4 12 36 108... 8 24 72 216... ... 写成这样的 ...
- 【BZOJ-2732】集合选数 状压DP (思路题)
2734: [HNOI2012]集合选数 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1070 Solved: 623[Submit][Statu ...
- 【BZOJ-2734】集合选数 状压DP (思路题)
2734: [HNOI2012]集合选数 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1070 Solved: 623[Submit][Statu ...
- bzoj2734【HNOI2012】集合选数
2734: [HNOI2012]集合选数 Time Limit: 10 Sec Memory Limit: 128 MB Submit: 831 Solved: 487 [Submit][Stat ...
- 【BZOJ2734】【HNOI2012】集合选数(状态压缩,动态规划)
[BZOJ2734][HNOI2012]集合选数(状态压缩,动态规划) 题面 Description <集合论与图论>这门课程有一道作业题,要求同学们求出{1, 2, 3, 4, 5}的所 ...
- BZOJ_2734_[HNOI2012]集合选数_构造+状压DP
BZOJ_2734_[HNOI2012]集合选数_构造+状压DP 题意:<集合论与图论>这门课程有一道作业题,要求同学们求出{1, 2, 3, 4, 5}的所有满足以 下条件的子集:若 x ...
- [HNOI2012]集合选数 --- 状压DP
[HNOI2012]集合选数 题目描述 <集合论与图论>这门课程有一道作业题,要求同学们求出\({1,2,3,4,5}\)的所有满足以 下条件的子集:若 x 在该子集中,则 2x 和 3x ...
- 状压DP之集合选数
题目 [HNOI2012]集合选数 <集合论与图论>这门课程有一道作业题,要求同学们求出{1, 2, 3, 4, 5}的所有满足以 下条件的子集:若 x 在该子集中,则 2x 和 3x 不 ...
随机推荐
- SSDB 使用笔记
1. SSDB中scan key_start key_end limit ,key_start 和 key_end 是指字母的顺序,不是数字. 2. 进入客户端:./ssdb-cli -p 8888
- redis 系列文章推荐
推荐博客: Redis在linux上的安装: http://www.open-open.com/lib/view/open1426468117367.html Redis的三种启动方式: http:/ ...
- Linux时间设置与iptables命令
日期与时间设置 timedatectl:显示目前时区与时间等信息 [root@localhost zhang]# timedatectl Local time: Thu 2018-01-18 10:1 ...
- 【Head First Servlets and JSP】笔记2:MVC迷你教程
1.采用MVC,不仅要求业务逻辑和表示分离.实际上,业务逻辑甚至根本不知道表示的存在.MVC的关键是,业务逻辑要与表示分离,而要求在中间放上别的东西,这样业务逻辑本身就可以作为一个可以重用的Java类 ...
- 一步一步教你读懂NET中IL
.NET CLR 和 Java VM 都是堆叠式虚拟机器(Stack-Based VM),也就是说,它们的指令集(Instruction Set)都是采用堆叠运算的方式:执行时的资料都是先放在堆叠中, ...
- qt项目: error LNK2038: 检测到“_MSC_VER”的不匹配项: 值“1900”不匹配值“1800”
error LNK2038: 检测到“_MSC_VER”的不匹配项: 值“1900”不匹配值“1800” 该错误 网上通常的解释是: 原因:由于你使用了vs2012,相比较vs2010以及之前的vs ...
- Go语言学习之运算符(The way to go)
生命不止,继续go go go 今天介绍go中的运算符. 运算符大概分为: Arithmetic Operators Relational Operators Logical Operators Bi ...
- [TJOI2010]打扫房间
题目描述 学校新建了一批宿舍,值日生小A要把所有的空房间都打扫一遍.这些宿舍的布局很奇怪,整个建筑物里所有的房间组成一个N * M的矩阵,每个房间的东南西北四面墙上都有一个门通向隔壁房间.另外有些房间 ...
- filebeat配置介绍
Filebeat Prospector filebeat.prospectors: - input_type: log paths: - /var/log/apache/httpd-*.log doc ...
- org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'sessionFactory' defined in class path resource [bean.xml]: Invocation of init method failed; nested exception is
在复制xml文件进行修改的时候,我经常将不小心对原文件进行修改,而导致创建bean出错.报错如下所示: Exception sending context initialized event to l ...