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的更多相关文章

  1. 2017CCPC秦皇岛 H题Prime Set&&ZOJ3988

    题意: 定义一种集合,只有两个数,两个数不同且加起来为素数.要从n个数里抽出数字组成该集合(数字也可以是1~n,这个好懵圈啊),要求你选择最多k个该种集合组成一个有最多元素的集合,求出元素的数量. 思 ...

  2. 【二分图最大匹配】【匈牙利算法】zoj3988 Prime Set

    题意:给你n个正整数,一对和为素数的数为一个合法数对.你选不超过K个合法数对,使得你选的数对涉及到的数的数量最大化.输出这个值. 所有1之间是可以任意两两配对的. 把奇数放在左侧,偶数放在右侧. 考虑 ...

  3. zoj3988 二分图匹配

    给一个数组,对于每两个数加起来为素数那么就是一个集合,求不超过k个集合的最多数是多少 解法:二分图匹配,先打素数筛,预处理边集,匹配完之后分两种情况k>匹配数,那么可以直接输出匹配数*2,否则可 ...

  4. zoj3988 Prime Set

    思路不难想到二分图求个最大匹配P,若P>=K,则2*K即可,否则应为P*2+min(K-P,未匹配且有度数不为0的顶点个数s).但坑点在于有1的情况,所以如果直接建二分图去跑最大匹配会因为1的影 ...

  5. ZOJ-3988 2017CCPC-秦皇岛 Prime Set 二分图最大匹配 匈牙利

    题面 题意:给你n个数,你可以选择2个和为质数的数为一对,每个数可以重复选择,你最多选k对,问你最多能选多少个不同数出来 题解:首先思考怎么样的数和为质数,2个偶数相加不行,除了1+1以外2个奇数相加 ...

随机推荐

  1. 什么是node.js的事件驱动编程

    Node.js现在非常活跃,相关生态社区已经超过Lua(基本上比较知名的功能都有nodejs模块实现).但是我们为何要使用Node.Js?相比传统的webserver服务模式,nodejs有什么优点优 ...

  2. git使用笔记(二)分支与合并

    By francis_hao    Nov 18,2016 查看分支,* 表示当前所在分支 $ git branch 查看分支和最后一次提交记录 $ git branch -v 新建分支 $ git ...

  3. json获取属性值的方式

    JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式.易于人阅读和编写.同时也易于机器解析和生成.它基于JavaScript(Standard ECMA-262 ...

  4. ntpq –p命令

    ntpq用来监视ntpd操作,ntpq -p查询网络中的NTP服务器,同时显示客户端和每个服务器的关系 [root@ ~]# ntpq -p remote refid st t when poll r ...

  5. 如何解决DuplicateFileException: Duplicate files copied in APK问题

    问题:有重复的文件存在APK里 解决方案:在Module里的build.gradle中设置忽略此重复文件即可.

  6. 链接oracle数据库 生成表对应的javabean

    package com.databi.utils; import java.io.File; import java.io.FileOutputStream; import java.io.IOExc ...

  7. tomcat:tomcat的OutOfMemoryError解决

    最近在熟悉一个开发了有几年的项目,需要把数据库从mysql移植到oracle,首先把jdbc的连接指向 mysql,打包放到tomcat里面,可以跑起来,没有问题,可是当把jdbc连接指向oracle ...

  8. macos装多个python

    简介 Mac包管理工具brew: 安装方法:命令行输入 /usr/bin/ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Ho ...

  9. 如何使主机和虚拟机IP处于同一网段(内网渗透专用)

    先说一下正常流程: 1.打开虚拟机网络设置选项,选择桥接模式(Bridged)[如果是Kali 2.0的话,执行第一步后就OK了(90%)] 2.打开Kali里面的网络设置 3.设置一个ip4或者ip ...

  10. bzoj 3196二逼平衡树 线段树套平衡树

    比较裸的树套树,对于区间K值bz上有一道裸题,详见题解http://www.cnblogs.com/BLADEVIL/p/3455336.html(其实题解也不是很详细) //By BLADEVIL ...