[ZOJ 3662] Math Magic (动态规划+状态压缩)
题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3662
之前写过这道题,结果被康神吐槽说代码写的挫。
的确,那时候想法太挫了。
题意:
给你三个数n,m,k。
问你存在多少个数列 a1,a2,...,ak,使得他们的和为n,他们的最小公倍数为m。
想法一:
因为 lcm(a1,a2,...,ak)=m,所以a1,a2,a3,...,ak都是m的约数。
因此预处理出来m的约数,记在w[i]里。
设计状态dp[i][j][k]代表有i个数的数列,和为j,最小公倍数为k的个数。
转移:dp[i][j][k] = sigma( dp[i-1][s][t] ) 其中 s+w[i] = j,lcm( t,w[i] ) = k
于是写成 dp[i][s+w[i]][lcm(t,w[i])] = sigma( dp[i-1][s][t] )
需要优化的部分较多,需要滚动数组,很容易TLE掉。
时间复杂度O( n*m*k*sqrt(n) )
于是被康神吐槽了。。
#include <bits/stdc++.h>
using namespace std; const int MOD = ;
int n,m,kk;
int a[],ptr;
int lcmn[][];
int dp[][][];
int ss[][][]; void getlcmn(){
for(int i=;i<;i++){
for(int j=;j<;j++){
lcmn[i][j] = i / __gcd(i,j) * j;
}
}
} int main(){
getlcmn();
while(scanf("%d%d%d",&n,&m,&kk)!=EOF){
ptr = ;
for(int i=;i*i<=m;i++){
if( m%i== ){
a[ptr++] = i;
if( i*i!=m ) a[ptr++] = m/i;
}
}
//for(int i=0;i<ptr;i++) printf("%d ",a[i]); puts("");
memset(dp,,sizeof(dp));
for(int i=;i<=min(m,n);i++) dp[][i][i] = ;
for(int i=;i<=kk;i++){
for(int j=;j<=n;j++){
for(int k=;k<=m;k++) dp[i&][j][k] = ;
for(int s=;s<ptr;s++){
if( j-a[s]>= ){
for(int t=;t<ptr;t++) {
if(lcmn[a[s]][a[t]]<=m) {
dp[i&][j][lcmn[a[s]][a[t]]] += dp[(i-)&][j-a[s]][a[t]];
dp[i&][j][lcmn[a[s]][a[t]]] %= MOD;
}
}
}
}
}
}
printf("%d\n",dp[kk&][n][m]);
}
return ;
}
想法一
当时听康神说所谓的正解,还真是没听懂,正巧昨天做了这道题:http://www.cnblogs.com/llkpersonal/p/4037686.html
于是今天早上就想起这个题目,也想了想“正解”。
由题意我们知道,m是a1,a2,...,ak的最小公倍数,因此,将a1,a2,a3,...,an质因数分解,取每一个质因数的最高次幂,然后乘起来就是m。
那么我们先对m进行质因数分解,然后划分状态,假设有sn个质因数,那么总状态数就是(1<<sn)-1个。
然后去分解m的因数,给每个因数定状态,如果说因数k的质因数的指数等于m的指数,那么就给该状态标1。记作v[i]
然后去跑完全背包。
定义状态dp[i][j][k][mask]代表数列有i个数,从前j个数里选,和为k,状态为mask。
状态转移:dp[i][j][k][mask] = sigma( dp[i-1][j-1][s][t] ) 其中s+w[i] = k , t|v[i]=mask
于是有dp[i][j][s+w[i]][t|v[i]] = sigma( dp[i-1][j-1][s][t] ) 对于每一个i,都相当于是一次独立的完全背包。
由于是完全背包,因此j维可以省掉。
于是化为
dp[j][s+w[i]][t|v[i]] = sigma( dp[j-1][s][t|v[i]] )
正向递推
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <cmath>
#include <map>
#include <iterator>
#include <vector>
using namespace std;
typedef long long LL; int n,m,k;
int p[],h[],w[],v[],ptr;
int dp[][][<<];
const int MOD = ; void prime_factor(int x){
for(int i=;i*i<=x;i++){
while( x%i== ){
p[i]++;
x /= i;
}
}
if( x!= ) p[x] = ;
} void handle(int x){
w[++ptr] = x;
for(int i=;i*i<=x;i++){
int t = ;
while( x%i== ){
t ++;
x /= i;
}
if( t&&t==p[i] ) v[ptr] |= (<<(h[i]-));
}
if( x!=&&p[x]== ) v[ptr] |= (<<(h[x]-));
} int main(){
while(scanf("%d%d%d",&n,&m,&k)!=EOF){
ptr = ;
memset(p,,sizeof(p));
memset(w,,sizeof(w));
memset(v,,sizeof(v));
memset(h,,sizeof(h));
prime_factor(m);
int sn = ;
for(int i=;i<;i++){
if( p[i] ) {
h[i] = ++sn;
}
}
for(int i=;i*i<=m;i++){
if( m%i== ) {
handle(i);
if( i*i!=m ) handle(m/i);
}
}
memset(dp,,sizeof(dp));
dp[][][] = ;
for(int j=;j<=k;j++){
for(int i=;i<=ptr;i++){
for(int k=;k<=n;k++){
for(int mask=;mask<(<<sn);mask++){
if( k+w[i]<=n&&(mask|v[i])<(<<sn) ){
dp[j][k+w[i]][mask|v[i]] = (dp[j][k+w[i]][mask|v[i]]+dp[j-][k][mask])%MOD;
}
}
}
}
}
printf("%d\n",dp[k][n][(<<sn)-]);
}
return ;
}
想法二
[ZOJ 3662] Math Magic (动态规划+状态压缩)的更多相关文章
- ZOJ 2563 Long Dominoes(状态压缩DP)
题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1563 题目大意:在h*w的矩阵里铺满1*3的小矩阵,共有多少种方法 ...
- [动态规划]状态压缩DP小结
1.小技巧 枚举集合S的子集:for(int i = S; i > 0; i=(i-1)&S) 枚举包含S的集合:for(int i = S; i < (1<<n); ...
- [POJ 2923] Relocation (动态规划 状态压缩)
题目链接:http://poj.org/problem?id=2923 题目的大概意思是,有两辆车a和b,a车的最大承重为A,b车的最大承重为B.有n个家具需要从一个地方搬运到另一个地方,两辆车同时开 ...
- POJ 1185 炮兵阵地(动态规划+状态压缩)
炮兵阵地 Description 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用"H" 表示),也可能是平原( ...
- ACM学习历程—HDU1584 蜘蛛牌(动态规划 && 状态压缩 || 区间DP)
Description 蜘蛛牌是windows xp操作系统自带的一款纸牌游戏,游戏规则是这样的:只能将牌拖到比她大一的牌上面(A最小,K最大),如果拖动的牌上有按顺序排好的牌时,那么这些牌也跟着一起 ...
- HDOJ-1074(动态规划+状态压缩)
Doing Homework HDOJ-1074 1.本题主要用的是状态压缩的方法,将每种状态用二进制压缩表示 2.状态转移方程:dp[i|(1<<j)]=min(dp[i|(1<& ...
- zoj 3471 Most Powerful(状态压缩dp)
Recently, researchers on Mars have discovered N powerful atoms. All of them are different. These ato ...
- 动态规划状态压缩-poj1143
题目链接:http://poj.org/problem?id=1143 题目描述: 代码实现: #include <iostream> #include <string.h> ...
- 3.4 熟练掌握动态规划——状态压缩DP
从旅行商问题说起—— 给定一个图,n个节点(n<=15),求从a节点出发,经历每个节点仅一次,最后回到a,需要的最短时间. 分析: 设定状态S代表当前已经走过的城市的集合,显然,S<=(1 ...
随机推荐
- JSP页面中<%! %>和<% %>的区别
JSP声明语句:<%!声明语句%>,通常声明全局变量.常量.方法.类JSP Scriptlet:<%java代码%>,其中可包含局部变量.java语句JSP表达式:<%= ...
- js中的各种宽高以及位置总结
在javascript中操作dom节点让其运动的时候,常常会涉及到各种宽高以及位置坐标等概念,如果不能很好地理解这些属性所代表的意义,就不能理解js的运动原理,同时,由于这些属性概念较多,加上浏览器之 ...
- Splashscreen
Splashscreen Enables developers to show/hide the application's splash screen. Methods show hide Perm ...
- LintCode "Submatrix Sum"
Naive solution is O(n^4). But on 1 certain dimension, naive O(n^2) can be O(n) by this well-known eq ...
- gdb: multiple process debug
gdbserver自身不支持multiple process:如果你调试parent process时在子进程上下断点,子进程在运行到那个断点时就会SIGTRAP. 如果你要调试fork出来的子进程: ...
- Linux下做软RAID
1.查看有多少块硬盘可用#fdisk -l嗯,一般而言,留下系统盘不动,其它的盘如果大小相同的话,统统合起来做一个raid. 2.决定做什么类型的raid.raid-0.raid-1.raid-5?如 ...
- memcache 永久数据被踢
1.memcache的slab chunk 就像公交车的坐位一样,有大小之分,还有活跃的状态. 2.代码 index.php -->插入数据,注意$value的大小和重复次数,跟chunk和其数 ...
- SecureCRT上传、下载文件 使用sz与rz命令
首先安装:apt-get install lrzsz SecureCRT这款SSH客户端软件同时具备了终端仿真器和文件传输功能.比ftp命令方便多了,而且服务器不用再开FTP服务了.rz,sz是便是L ...
- 【转】c# winform DataGridView导出数据到Excel中,可以导出当前页和全部数据
准备工作就是可以分页的DataGridView,和两个按钮,一个用来导出当前页数据到Excel,一个用来导出全部数据到Excel 没有使用SaveFileDialog,但却可以弹出保存对话框来 先做导 ...
- Top 6 Programming Languages for Mobile App Development
Mobile application development industry in the last five years have multiplied in leaps and bounds, ...