zoj3988
zoj3988
题意
如果一个集合 \(\{i,j\}\) 满足 \(i\neq j\) 且 \(a[i]+a[j]\) 是素数,则称之为素数集合。
给出一些数字,这些数字可以组成多种素数集合,从这些集合中最多选择 \(k\) 个集合,问这些集合涉及到的数的数量最大值为多少。
分析
存在匹配关系即 \(a[i]+a[j]\) 是素数,那么 \(i\) \(j\) 就可以连边,要求两个数的和为素数,那么这两个数一定有一奇数一偶数(也有可能两个 \(1\)),将奇数放左边,偶数放右边,建二分图。求一发二分图匹配即可。
要注意的是两个 1 的和也是素数,首先奇数 1 放在最后不急着匹配,如果 \(1\) 的数量大于 \(1\) 的话,先将两个 \(1\) 组合起来。
code
#include<bits/stdc++.h>
typedef long long ll;
using namespace std;
const int N = 2e6 + 10;
const int MAXN = 3e3 + 5;
int notprime[N];
int n, k;
vector<int> odd, even;
int vis[MAXN], has[MAXN], a[MAXN], b[MAXN];
vector<int> v[MAXN];
int dfs(int x) {
for(int i = 0; i < v[x].size(); i++) {
int to = v[x][i];
if(!vis[to]) {
vis[to] = 1;
if(has[to] == -1 || dfs(has[to])) {
has[to] = x;
return 1;
}
}
}
return 0;
}
int main() {
for(int i = 2; i < N; i++) if(!notprime[i]) {
for(ll j = 1LL * i * i; j < N; j += i) notprime[j] = 1;
}
int T;
scanf("%d", &T);
while(T--) {
odd.clear();
even.clear();
memset(has, -1, sizeof has);
scanf("%d%d", &n, &k);
int cnt = 0;
for(int i = 0; i < n; i++) {
int x;
scanf("%d", &x);
if(x == 1) cnt++;
else if(x & 1) odd.push_back(x);
else even.push_back(x);
}
int cc = cnt;
while(cc--) odd.push_back(1);
int ans = 0;
for(int i = 0; i < odd.size(); i++) {
v[i].clear();
for(int j = 0; j < even.size(); j++) {
if(!notprime[odd[i] + even[j]]) {
v[i].push_back(j);
}
}
memset(vis, 0, sizeof vis);
if(dfs(i) && k) {
ans += 2;
k--;
}
}
memset(a, 0, sizeof a);
memset(b, 0, sizeof b);
for(int i = 0; i < even.size(); i++) {
if(has[i] != -1) {
a[has[i]] = 1;
b[i] = 1;
}
}
int cnt1 = 0;
for(int i = 0; i < odd.size() && k; i++) {
if(!a[i] && odd[i] == 1) cnt1++;
}
while(k && cnt1 > 1) {
cnt1 -= 2;
ans += 2;
k--;
}
int flg = cnt1;
for(int i = 0; i < odd.size() && k; i++) {
if(!a[i] && odd[i] == 1) {
if(flg) {
flg--;
} else {
a[i] = 1;
}
}
}
for(int i = 0; i < odd.size() && k; i++) {
for(int j = 0; j < v[i].size() && k; j++) {
int to = v[i][j];
if(a[i] && !b[to]) { ans++; k--; b[to] = 1; }
else if(!a[i] && b[to]) { ans++; k--; if(odd[i] == 1) cnt1--; a[i] = 1; }
}
}
if(k && cnt > 1 && cnt1) ans++;
printf("%d\n", ans);
}
return 0;
}
zoj3988的更多相关文章
- 2017CCPC秦皇岛 H题Prime Set&&ZOJ3988
题意: 定义一种集合,只有两个数,两个数不同且加起来为素数.要从n个数里抽出数字组成该集合(数字也可以是1~n,这个好懵圈啊),要求你选择最多k个该种集合组成一个有最多元素的集合,求出元素的数量. 思 ...
- 【二分图最大匹配】【匈牙利算法】zoj3988 Prime Set
题意:给你n个正整数,一对和为素数的数为一个合法数对.你选不超过K个合法数对,使得你选的数对涉及到的数的数量最大化.输出这个值. 所有1之间是可以任意两两配对的. 把奇数放在左侧,偶数放在右侧. 考虑 ...
- zoj3988 二分图匹配
给一个数组,对于每两个数加起来为素数那么就是一个集合,求不超过k个集合的最多数是多少 解法:二分图匹配,先打素数筛,预处理边集,匹配完之后分两种情况k>匹配数,那么可以直接输出匹配数*2,否则可 ...
- zoj3988 Prime Set
思路不难想到二分图求个最大匹配P,若P>=K,则2*K即可,否则应为P*2+min(K-P,未匹配且有度数不为0的顶点个数s).但坑点在于有1的情况,所以如果直接建二分图去跑最大匹配会因为1的影 ...
- ZOJ-3988 2017CCPC-秦皇岛 Prime Set 二分图最大匹配 匈牙利
题面 题意:给你n个数,你可以选择2个和为质数的数为一对,每个数可以重复选择,你最多选k对,问你最多能选多少个不同数出来 题解:首先思考怎么样的数和为质数,2个偶数相加不行,除了1+1以外2个奇数相加 ...
随机推荐
- win10 update orchestratorservere禁用
1 Windows 10系统中有一项Update Orchestrator Service(更新协调器办事),在当地办事窗口中,我们发现 Update Orchestrator Service 办 ...
- JavaScript的大括号的语义
Javascript中大括号"{}"有四种语义作用: 语义1. 组织复合语句,这是最常见的: view source print? 1 if( condition ) { 2 ...
- Codeforces Round #524 (Div. 2) B. Margarite and the best present
B. Margarite and the best present 题目链接:https://codeforces.com/contest/1080/problem/B 题意: 给出一个数列:an=( ...
- oracle的rownum使用
对于rownum来说它是Oracle系统顺序分配为从查询返回的行的编号,返回的第一行分配的是1,第二行是2,依此类推,这个伪字段可以用于限制查询返回的总行数,且rownum不能以任何表的名称作为前缀. ...
- Out of memory due to hash maps used in map-side aggregation解决办法
在运行一个group by的sql时,抛出以下错误信息: Task with the most failures(4): -----Task ID: task_201411191723_723592 ...
- 数据结构之DFS与BFS
深度搜索(DFS) and 广度搜索(BFS) 代码如下: #include "stdafx.h" #include<iostream> #include<st ...
- noip车站分级 拓扑排序
题目传送门 这道题呢 每次输入一段数就把1~n里面没有在这组数里面的数和他们连一波 表示这些数比他们等级低 然后就搞一搞就好了哇 #include<cstdio> #include< ...
- DP+贪心水题合集_C++
本文含有原创题,涉及版权利益问题,严禁转载,违者追究法律责任 本次是最后一篇免费的考试题解,以后的考试题目以及题解将会以付费的方式阅读,题目质量可以拿本次作为参考 本来半个月前就已经搞得差不多了,然后 ...
- bzoj 1066 最大流
将每个石柱拆成两个点,分别是进入的和出去的,两个点之间连石柱的高度 然后每个出去的点连别的石柱的进去的点, 源点连所有蜥蜴所在柱子,每个能跳出去的连汇点,然后最大流就行了 /************* ...
- UVALIVE 3486 Cells
通过入栈出栈顺序判断祖先关系 这里UVALIVE还 #include <map> #include <set> #include <list> #include & ...