Professional layer CodeForces - 1103D (状压,gcd)
大意: 给定$n$元素序列$a$, 现在想要让$gcd(a_1,a_2,...,a_n)=1$. 对于每个$a_i$可以除以一个不超过$k$的因子, 代价为$e_i$, 假设一共选择了$x$个元素去除, 代价和为$y$, 求$xy$的最小值.
设$g=gcd(a_1,a_2,...,a_n)=p_1^{\alpha_1}p_2^{\alpha_2}\cdots p_r^{\alpha_r}$, 有$1\le r \le 11$
我们考虑最优解的结构, 对于一个数$a_i$, 它的某个素因子$p_k$想要对答案产生贡献则必须将$p_k$全部除去, 所以每个数都可以用一个二进制状态表示, 并且显然最优解至多选择$r$个$a_i$. 这样的话就可以得到暴力代码, 转移复杂度是$O(nr2^{2r})$, 显然过不去
//dp[i][j][k] 表示前i个数取j个数状态为k时的最小值
memset(dp,0x3f,sizeof dp);
dp[0][0][0]=0;
REP(i,1,n) {
//预处理出每个状态需要除的数num
REP(j,1,S) if (num[j]<=k) {
REP(x,0,S) {
REP(t,1,r) {
chkmin(dp[i][t][x|j],dp[i-1][t-1][x]+a[i].e);
}
}
}
}
ll ans = INF;
REP(i,1,r) if (dp[n][i][S]!=INF) ans=min(ans, dp[n][i][S]*i);
下面考虑优化
- 对于每个$a_i$, 对于它不在$g$内的素因子直接除去不考虑, 打表发现这样不同的$a_i$不超过M=12000
- 对于每个不同的$a_i$只取$e$前$r$小的, 并且每个状态只至多转移$r$次
- 枚举补集的子集, 可将$2^{2r}$优化为$3^r$
这样转移的复杂度就为$O(Mr2^r+r^23^r)$
最后要注意最大素因子是$O(max{a_i})$的, 要开long long 存!!
#include <iostream>
#include <algorithm>
#include <cstdio>
#include <math.h>
#include <set>
#include <map>
#include <queue>
#include <string>
#include <string.h>
#define REP(i,a,n) for(int i=a;i<=n;++i)
#define PER(i,a,n) for(int i=n;i>=a;--i)
#define hr putchar(10)
#define pb push_back
#define lc (o<<1)
#define rc (lc|1)
#define mid ((l+r)>>1)
#define ls lc,l,mid
#define rs rc,mid+1,r
#define x first
#define y second
#define io std::ios::sync_with_stdio(false)
#define endl '\n'
using namespace std;
typedef long long ll;
typedef pair<int,int> pii;
const int P = 1e9+7;
const ll INF = 0x3f3f3f3f3f3f3f3f;
ll gcd(ll a,ll b) {return b?gcd(b,a%b):a;}
ll qpow(ll a,ll n) {ll r=1%P;for (a%=P;n;a=a*a%P,n>>=1)if(n&1)r=r*a%P;return r;}
ll inv(ll x){return x<=1?1:inv(P%x)*(P-P/x)%P;}
//head const int N = 1e6+10; int n, w[N];
ll k, g;
struct _ {
ll a;
int e;
bool operator < (const _ &rhs) const {
return e<rhs.e;
}
} a[N];
vector<ll> A;
map<ll,int> vis;
ll dp[2][12][1<<11], num[1<<11]; void factor(ll g) {
int mx = sqrt(g+0.5);
REP(i,2,mx) if (g%i==0) {
A.pb(i);
while (g%i==0) g/=i;
}
if (g>1) A.pb(g);
} void chkmin(ll &x, ll y) {x=min(x,y);} int main() {
scanf("%d%lld", &n, &k);
REP(i,1,n) scanf("%lld", &a[i].a),g=gcd(g,a[i].a);
if (g==1) return puts("0"),0;
REP(i,1,n) scanf("%d", &a[i].e);
sort(a+1,a+1+n);
factor(g);
int r = A.size(), S = (1<<r)-1, cur = 0;
memset(dp[0],0x3f,sizeof dp[0]);
dp[0][0][0]=0;
REP(i,1,n) {
ll b = 1;
REP(j,0,r-1) if (a[i].a%A[j]==0) {
ll t = 1;
while (a[i].a%A[j]==0) a[i].a/=A[j],t*=A[j];
num[1<<j] = t;
b *= t;
}
if (++vis[b]>r) continue;
num[0] = 1;
cur ^= 1;
memcpy(dp[cur],dp[cur^1],sizeof dp[cur]);
REP(j,1,S) {
num[j] = num[j^j&-j]*num[j&-j];
if (num[j]>k) continue;
if (++vis1[j]>r) continue;
ll x = ~j&S;
PER(t,0,r-1) {
for (int y=x; y; y=(y-1)&x) if (dp[cur^1][t][y]!=INF) {
chkmin(dp[cur][t+1][y^j],dp[cur^1][t][y]+a[i].e);
}
chkmin(dp[cur][t+1][j],dp[cur^1][t][0]+a[i].e);
}
}
}
ll ans = INF;
REP(i,1,r) if (dp[cur][i][S]!=INF) {
ans = min(ans, dp[cur][i][S]*i);
}
printf("%lld\n", ans==INF?-1:ans);
}
Professional layer CodeForces - 1103D (状压,gcd)的更多相关文章
- Vladik and cards CodeForces - 743E (状压)
大意: 给定序列, 求选出一个最长的子序列, 使得任选两个[1,8]的数字, 在子序列中的出现次数差不超过1, 且子序列中相同数字连续. 正解是状压dp, 先二分转为判断[1,8]出现次数>=x ...
- Keyboard Purchase CodeForces - 1238E (状压)
大意: 给定串$s$, 字符集为字母表前$m$个字符, 求一个$m$排列$pos$, 使得$\sum\limits_{i=2}^n|{pos}_{s_{i-1}}-{pos}_{s_{i}}|$最小. ...
- CodeForces 11D(状压DP 求图中环的个数)
Given a simple graph, output the number of simple cycles in it. A simple cycle is a cycle with no re ...
- Clear The Matrix CodeForces - 903F (状压)
大意: 给定4行的棋盘以及4种大小的正方形方块, 每种各有一定花费, 每次可以选一种方块放在棋盘上, 棋盘对应格子全变为'.', 求最少花费使得棋盘全部变成'.' 状压基本操作练习, 状态取12位, ...
- Pollywog CodeForces - 917C (状压)
链接 大意: 一共n个格子, 初始$x$只蝌蚪在前$x$个格子, 每次最左侧的蝌蚪向前跳, 跳跃距离在范围[1,k], 并且每只蝌蚪跳跃都有一定花费, 有$q$个格子上有石头, 若有蝌蚪跳到某块石头上 ...
- Codeforces 678E 状压DP
题意:有n位选手,已知n位选手之间两两获胜的概率,问主角(第一个选手)最终站在擂台上的概率是多少? 思路:一看数据范围肯定是状压DP,不过虽然是概率DP,但是需要倒着推:我们如果正着推式子的话,初始状 ...
- Codeforces 8C 状压DP
题意:有个人想收拾行李,而n个物品散落在房间的各个角落里(n < 24).现在给你旅行箱的坐标(人初始在旅行箱处),以及n个物品的坐标,你一次只能拿最多两个物品,并且拿了物品就必须放回旅行箱,不 ...
- Codeforces 1215E 状压DP
题意:给你一个序列,你可以交换序列中的相邻的两个元素,问最少需要交换多少次可以让这个序列变成若干个极大的颜色相同的子段. 思路:由于题目中的颜色种类很少,考虑状压DP.设dp[mask]为把mask为 ...
- Crisp String CodeForces - 1117F (状压)
#include <iostream> #include <algorithm> #include <cstdio> #include <math.h> ...
随机推荐
- 2018-2019-2 20165209 《网络对抗技术》Exp4:恶意代码分析
2018-2019-2 20165209 <网络对抗技术>Exp4:恶意代码分析 1 基础问题回答和实验内容 1.1基础问题回答 如果在工作中怀疑一台主机上有恶意代码,但只是猜想,所有想监 ...
- WireShark 基本介绍
文中内容主要转自:http://www.cnblogs.com/TankXiao/archive/2012/10/10/2711777.html 一.Wireshark 与 Fiddler 比较: F ...
- Linux基础命令---bc
bc bc是一种算数语言,其语法和c语言类似,可以交互执行.通过命令行选项可以获得一个标准的数学库.如果请求,在处理任何文件之前定义数学库.BC从处理所有文件的代码开始.命令行中列出的文件按所列顺序排 ...
- MySQL索引类型总结和使用技巧
引用地址:http://www.jb51.net/article/49346.htm 在数据库表中,对字段建立索引可以大大提高查询速度.假如我们创建了一个 mytable表: 复制代码 代码如下: C ...
- cache与buffer
Cache 缓存区,是高速缓存,是位于CPU和主内存之间的容量较小但速度很快的存储器,因为CPU的速度远远高于主内存的速度,CPU从内存中读取数据需等待很长的时间,而 Cache保存着CPU刚用过的 ...
- lambda表达式Bug——修改捕获变量失败
解<C++ Primer 5th>的 9-50 练习题时,遇到了 lambda表达式值捕获和引用捕获之区别问题. 欲修改捕获的变量 sum,累加之.但当时忘记值捕获和引用捕获是有区别的.下 ...
- JAVA I/O(一)基本字节和字符IO流
最近再看I/O这一块,故作为总结记录于此.JDK1.4引入NIO后,原来的I/O方法都基于NIO进行了优化,提高了性能.I/O操作类都在java.io下,大概将近80个,大致可以分为4类: 基于字节操 ...
- dll和ocx的简单理解
一.dll dll就是打包一些程序或者算法,根据我的理解分个类 1.算法的打包 比如打包C/C++的一些纯代码算法,计算平均值,极值,标准差....,只需要向外提供接口和入口参数,外部即可轻松调用 2 ...
- Mac升级到EI Captain之后pip install 无法使用问题
错误log: creating /System/Library/Frameworks/Python.framework/Versions/2.7/share error: could not crea ...
- WPF基础学习笔记整理 (七) Binding绑定
基础知识: 数据绑定是一种关系,该关系告诉WPF从源对象提取一些信息,并用这些信息设置目标对象的属性:目标对象始终是依赖属性,而源对象则可以是任何内容. BindingOperations类,提供静态 ...