BZOJ4514——[Sdoi2016]数字配对
#include <map> #include <set> #include <queue> #include <cmath> #include <cstdio> #include <cstdlib> #include <cstring> #include <algorithm> using namespace std; #define LL long long #define llinf 2000000000000000000 #define inf 2147483647 #define for1(i, x, y) for(LL i = (x); i <= (y); i ++) #define for2(i, x, y) for(LL i = (x); i >= (y); i --) namespace mcmf{ LL s, t; struct Edge{ LL u, v, cap, flow; LL cost; LL next; } G[250010]; LL tot; LL head[3000]; LL inq[3000]; LL d[3000]; LL p[3000]; LL a[3000]; inline void init(){ memset(head, -1, sizeof(head)); tot = -1; } inline void add(LL u, LL v, LL w, LL cost){ G[++ tot] = (Edge){u, v, w, 0, cost, head[u]}; head[u] = tot; G[++ tot] = (Edge){v, u, 0, 0, -cost, head[v]}; head[v] = tot; return; } inline bool BellmanFord(LL& flow, LL& cost){ for(LL i = s; i <= t; i ++) d[i] = -llinf; memset(inq, 0, sizeof(inq)); d[s] = 0; inq[s] = 1; p[s] = 0; a[s] = inf; queue<LL> Q; Q.push(s); while(!Q.empty()){ LL u = Q.front(); Q.pop(); inq[u] = 0; for(LL i = head[u]; i != -1; i = G[i].next){ Edge& e = G[i]; if(e.cap > e.flow && d[e.v] < d[u] + e.cost){ d[e.v] = d[u] + e.cost; p[e.v] = i; a[e.v] = min(a[u], e.cap - e.flow); if(!inq[e.v]){ Q.push(e.v); inq[e.v] = 1; } } } } if(d[t] == -llinf) return false; flow += a[t]; cost += d[t] * (LL)a[t]; if(cost < 0){ cost -= d[t] * (LL)a[t]; flow -= a[t]; flow += cost / -d[t]; return false; } LL u = t; while(u != s){ G[p[u]].flow += a[t]; G[p[u] ^ 1].flow -= a[t]; u = G[p[u]].u; } return true; } inline LL Minflow(){ LL flow = 0; LL cost = 0; while(BellmanFord(flow, cost)); return flow; } } LL a[100010], b[100010], c[100010]; LL prime[100010], tot; bool vis[100010]; LL cnta[100010]; inline LL read(){ // getchar较快于scanf,更快的还有fread,不会23333 char ch = getchar(); LL x = 0, f = 1; while(ch < '0' || ch > '9'){ if(ch == '-') f = -1; ch = getchar(); } while('0' <= ch && ch <= '9'){ x = x * 10 + ch - '0'; ch = getchar(); } return x * f; } inline LL llread(){ char ch = getchar(); LL x = 0, f = 1; while(ch < '0' || ch > '9'){ if(ch == '-') f = -1; ch = getchar(); } while('0' <= ch && ch <= '9'){ x = x * 10 + ch - '0'; ch = getchar(); } return x * f; } inline void init_prime(){ for1(i, 2, 100000){ if(!vis[i]) prime[++ tot] = i; for1(j, 1, tot){ if(i * prime[j] > 100000) break; vis[i * prime[j]] = 1; if(i % prime[j] == 0) break; } } } inline bool is_prime(LL x){ if(x <= 100000) return 1 - vis[x]; LL t = sqrt(x); for1(i, 1, tot){ if(prime[i] > t) break; if(x % prime[i] == 0) return false; } return true; } int main(){ LL n = read(); for1(i, 1, n) a[i] = llread(); for1(i, 1, n) b[i] = llread(); for1(i, 1, n) c[i] = llread(); mcmf::init(); init_prime(); for1(i, 1, n){ LL t = sqrt(a[i]); LL o = a[i]; for1(j, 1, tot){ if(prime[j] > t) break; while(o % prime[j] == 0) o /= prime[j], cnta[i] ++; if(o == 1) break; } if(o != 1) cnta[i] ++; } mcmf::s = 0; mcmf::t = n + 1; for1(i, 1, n){ if(cnta[i] & 1) mcmf::add(0, i, b[i], 0); else mcmf::add(i, n + 1, b[i], 0); } for1(i, 1, n) if(cnta[i] & 1){ for1(j, 1, n) if(!(cnta[j] & 1)){ if(a[i] % a[j] != 0 && a[j] % a[i] != 0) continue; if(a[i] % a[j] == 0 && is_prime(a[i] / a[j])) mcmf::add(i, j, inf, c[i] * c[j]); else if(a[j] % a[i] == 0 && is_prime(a[j] / a[i])) mcmf::add(i, j, inf, c[i] * c[j]); } } printf("%lld", mcmf::Minflow()); return 0; }
BZOJ4514——[Sdoi2016]数字配对的更多相关文章
- bzoj4514 [Sdoi2016]数字配对
Description 有 n 种数字,第 i 种数字是 ai.有 bi 个,权值是 ci. 若两个数字 ai.aj 满足,ai 是 aj 的倍数,且 ai/aj 是一个质数, 那么这两个数字可以配对 ...
- BZOJ4514[Sdoi2016]数字配对——最大费用最大流
题目描述 有 n 种数字,第 i 种数字是 ai.有 bi 个,权值是 ci. 若两个数字 ai.aj 满足,ai 是 aj 的倍数,且 ai/aj 是一个质数, 那么这两个数字可以配对,并获得 ci ...
- bzoj4514 [Sdoi2016]数字配对(网络流)
Description 有 n 种数字,第 i 种数字是 ai.有 bi 个,权值是 ci. 若两个数字 ai.aj 满足,ai 是 aj 的倍数,且 ai/aj 是一个质数, 那么这两个数字可以配对 ...
- [bzoj4514][SDOI2016]数字配对——二分图
题目描述 传送门 题解: 这个题真的是巨坑,经过了6个WA,2个TLE,1个RE后才终于搞出来,中间都有点放弃希望了... 主要是一定要注意longlong! 下面开始说明题解. 朴素的想法是: 如果 ...
- BZOJ4514 [Sdoi2016]数字配对 【费用流】
题目 有 n 种数字,第 i 种数字是 ai.有 bi 个,权值是 ci. 若两个数字 ai.aj 满足,ai 是 aj 的倍数,且 ai/aj 是一个质数, 那么这两个数字可以配对,并获得 ci×c ...
- bzoj4514: [Sdoi2016]数字配对--费用流
看了一眼题目&数据范围,觉得应该是带下界的费用流 原来想拆点变成二分图,能配对的连边,跑二分图,可行性未知 后来看到另外一种解法.. 符合匹配要求的数要满足:质因子的个数相差为1,且两者可整除 ...
- bzoj4514: [Sdoi2016]数字配对(费用流)
传送门 ps:费用流增广的时候费用和流量打反了……调了一个多小时 每个数只能参与一次配对,那么这就是一个匹配嘛 我们先把每个数分解质因数,记质因子总个数为$cnt_i$,那如果$a_i/a_j$是质数 ...
- 【bzoj4514】: [Sdoi2016]数字配对 图论-费用流
[bzoj4514]: [Sdoi2016]数字配对 好像正常的做法是建二分图? 我的是拆点然后 S->i cap=b[i] cost=0 i'->T cap=b[i] cost=0 然后 ...
- 【BZOJ4514】[Sdoi2016]数字配对 费用流
[BZOJ4514][Sdoi2016]数字配对 Description 有 n 种数字,第 i 种数字是 ai.有 bi 个,权值是 ci. 若两个数字 ai.aj 满足,ai 是 aj 的倍数,且 ...
随机推荐
- 深入JVM-性能监控工具
一.Linux下的性能监控工具 1.1 显式系统整体资源使用情况-top命令 top命令的输出可以分为两个部分:前半部分是系统统计信息,后半部分是进程信息. 在统计信息中,第一行是任务队列信息,他的结 ...
- ARCGIS9.2安装说明
1. 如果已经安装 ArcGIS 9.1, 先完全卸载. 2. License Manger 安装 a. 创建此路径: ...
- 20145212 《Java程序设计》第6周学习总结
20145212 <Java程序设计>第6周学习总结 学习内容总结 第十章 1.Java将输入/输出抽象化为串流.读取数据对象成为输入流,能向其写入的对象叫输出流. 我从网上找到了一个可以 ...
- JavaScript学习笔记——对表单的操作
javascript-对表单的操作实例讲解 <form name="myform" id="form1" action="" meth ...
- 升级10.11.6后CocoaPods的坑,之前10.11.4已经安装好的,居然没了Failed to locate Homebrew!
升级10.11.6后CocoaPods的坑,之前10.11.4已经安装好的,居然没了,用命令 sudo gem install cocoapod 装不上,换 sudo gem install -n/u ...
- Unity 下载
Unity历史版本 http://wiki.ceeger.com/unity:history#unity_522f1 UNITY 下载存档 http://unity3d.com/cn/get-unit ...
- Redis-cluster集群【第四篇】:redis-cluster集群配置
Redis分片: 为什么要分片:随着Redis存储的数据越来越庞大,会导致Redis的性能越来越差! 目前分片的方法: 1.客户端分片 在应用层面分片,程序里指定什么数据存放在那个Redis 优势: ...
- Python之路【第三篇】:Python基础(二)
函数的理解 面向过程:根据业务逻辑从上到下写垒代码 函数式:将某功能代码封装到函数中,日后便无需重复编写,仅调用函数即可 函数作用是你的程序有良好的扩展性.复用性. 同样的功能要是用3次以上的话就建议 ...
- Struts2源码浅析-ConfigurationProvider
ConfigurationProvider接口 主要完成struts配置文件 加载 注册过程 ConfigurationProvider接口定义 public interface Configurat ...
- 从HTML原型到jsp页面完美转型攻略(教你即使不会写代码也能弄出漂亮的网页)
大家都知道软件项目(web)开发之前都要先做原型设计,而我们使用的比较多的一款原型设计软件就是Axure rp了.在Axure rp上画原型不需要任何编码能力,而且生成的原型可以在浏览器上运行.除了没 ...