题目描述
有 n 种数字,第 i 种数字是 ai、有 bi 个,权值是 ci。
若两个数字 ai、aj 满足,ai 是 aj 的倍数,且 ai/aj 是一个质数,
那么这两个数字可以配对,并获得 ci×cj 的价值。
一个数字只能参与一次配对,可以不参与配对。
在获得的价值总和不小于 0 的前提下,求最多进行多少次配对。
分析
建图比较裸,类似于二分图的建图方法即可
需要注意的是,要求在获得的价值总和不小于 0 的前提下进行配对,所以我们spfa每次跑最长路,如果这一次spfa的产生的费用会导致总费用变成负数,我们就要贪心的去想我们最多可以完成几次配对即可
代码
#pragma GCC optimize(3)
#include <bits/stdc++.h>
#define debug(x) cout<<#x<<":"<<x<<endl;
#define dl(x) printf("%lld\n",x);
#define di(x) printf("%d\n",x);
#define _CRT_SECURE_NO_WARNINGS
#define pb push_back
#define mp make_pair
#define all(x) (x).begin(),(x).end()
#define fi first
#define se second
#define SZ(x) ((int)(x).size())
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int, int> PII;
typedef vector<int> VI;
const ll INF = 0x3f3f3f3f3f3f3f3f;
const int N = 220 * 2,M = N * N * 2;
const ll mod = 1000000007;
const double eps = 1e-9;
const double PI = acos(-1);
template<typename T>inline void read(T &a) {
char c = getchar(); T x = 0, f = 1; while (!isdigit(c)) {if (c == '-')f = -1; c = getchar();}
while (isdigit(c)) {x = (x << 1) + (x << 3) + c - '0'; c = getchar();} a = f * x;
}
int gcd(int a, int b) {return (b > 0) ? gcd(b, a % b) : a;}
int h[N],ne[M],e[M],idx;
ll w[M],d[N],incf[N],f[M];
int n,m,S,T;
int a[N],b[N];
ll c[N];
int q[N],pre[N];
bool st[N];
void add(int a,int b,ll c,ll d){
ne[idx] = h[a],e[idx] = b,f[idx] = c,w[idx] = d,h[a] = idx++;
ne[idx] = h[b],e[idx] = a,f[idx] = 0,w[idx] = -d,h[b] = idx++;
}
bool spfa(){
int hh = 0,tt = 1;
memset(d,-0x3f,sizeof d);
memset(incf,0,sizeof incf);
q[0] = S,d[S] = 0,incf[S] = INF;
while(hh != tt){
int t = q[hh++];
if(hh == N) hh = 0;
st[t] = false;
for(int i = h[t];~i;i = ne[i]){
int ver = e[i];
if(f[i] && d[ver] < d[t] + w[i]){
pre[ver] = i;
d[ver] = d[t] + w[i];
incf[ver] = min(f[i],incf[t]);
if(!st[ver]){
q[tt++] = ver;
if(tt == N) tt = 0;
st[ver] = true;
}
}
}
}
return incf[T] > 0;
}
void EK(ll &flow,ll &cost){
flow = cost = 0;
while(spfa()){
ll t = incf[T];
if(cost + t * d[T] >= 0) cost += t * d[T],flow += t;
else {
flow += cost / (-d[T]);
return;
}
for(int i = T;i != S;i = e[pre[i] ^ 1]){
f[pre[i]] -= t;
f[pre[i] ^ 1] += t;
}
}
}
bool isprime(int x){
for(int i = 2;i <= sqrt(x);i++) if(x % i == 0) return true;
return false;
}
int main() {
memset(h,-1,sizeof h);
read(n);
for(int i = 1;i <= n;i++) read(a[i]);
for(int i = 1;i <= n;i++) read(b[i]);
for(int i = 1;i <= n;i++) read(c[i]);
S = 0,T = N - 1;
for(int i = 1;i <= n;i++) add(S,i,b[i],0);
for(int i = n + 1;i <= 2 * n;i++) add(i,T,b[i - n],0);
for(int i = 1;i <= n;i++)
for(int j = 1;j <= n;j++){
if(a[i] % a[j] || a[i] / a[j] <= 1) continue;
if(isprime(a[i] / a[j])) continue;
add(i,n + j,100000000,c[i] * c[j]);
add(j,i + n,100000000,c[i] * c[j]);
}
ll flow = 0,cost = 0;
EK(flow,cost);
dl(flow / 2);
return 0;
}
/**
* ┏┓ ┏┓+ +
* ┏┛┻━━━┛┻┓ + +
* ┃ ┃
* ┃ ━ ┃ ++ + + +
* ████━████+
* ◥██◤ ◥██◤ +
* ┃ ┻ ┃
* ┃ ┃ + +
* ┗━┓ ┏━┛
* ┃ ┃ + + + +Code is far away from
* ┃ ┃ + bug with the animal protecting
* ┃ ┗━━━┓ 神兽保佑,代码无bug
* ┃ ┣┓
* ┃ ┏┛
* ┗┓┓┏━┳┓┏┛ + + + +
* ┃┫┫ ┃┫┫
* ┗┻┛ ┗┻┛+ + + +
*/