P3940 分组
P3940 分组
https://www.luogu.org/problemnew/show/P3940
官方题解http://pan.baidu.com/s/1eSAMuXk
分析:
并查集。
首先根据K=1和K=2分成两个问题来做。
K=1:问题为分成最小数量的区间,使得每个区间满足:任意两个数的和都不是完全平方数(1,3位于一个集合是不可以的),输出字典序最小的方案。
一个性质:一个集合越长越好(ppt里有证明)。那么从后往前扫,每到一个判断是否可以加进去,不可以的时候,记录答案。如果直接判断是$n^2$,可以枚举$x^2$来判断,复杂度$O(n\sqrt n)$。
K=2:问题为分成最小数量的区间,使得每个区间满足:区间可以任意划分为至多两个集合,这两个集合内部任意两个数的和都不是完全平方数。(1,3,5是可以在分成一段的,因为可以划分为{1,3},{5})
上面的性质还是适用的,所以同样是从后往前扫,每扫到一个判断是否可以加入。这个地方可以记录一个并查集,维护“敌人”集合。$x+n$表示x的“敌人”(不可以一起存在的)。那么如果两个数a+b属于同一个集合,说明a和b用一个共同的敌人,那么a和b就必须在同一个集合里。加入一个x,判断是否可以$k^2-x$共存在这个区间里。注意到一个问题:如果一个数出现了多次,如果它是$2x=k^2$的形式,需要特判(见代码),其他的都可以放进一个集合里。
代码:
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<iostream>
#include<cctype>
#include<set>
#include<vector>
#include<queue>
#include<map>
#define fi(s) freopen(s,"r",stdin);
#define fo(s) freopen(s,"w",stdout);
using namespace std;
typedef long long LL; inline int read() {
int x=,f=;char ch=getchar();for(;!isdigit(ch);ch=getchar())if(ch=='-')f=-;
for(;isdigit(ch);ch=getchar())x=x*+ch-'';return x*f;
} const int N = ; int a[N], fa[N << ];
int n, Mx;
bool vis[N], Ban[N], issqr[N << ];
vector<int> ans; void solve1() {
for (int j = n, i = n; i>=; ) {
for (bool flag = ; j >= ; -- j) {
for (int k=, tmp; ; ++ k) {
tmp = k * k - a[j];
if (tmp <= ) continue; if (tmp > Mx) break;
if (vis[tmp]) { flag = ; break; }
}
if (!flag) break;
vis[a[j]] = true;
}
if (!j) break;
ans.push_back(j);
for (; i > j; -- i) vis[a[i]] = false;
}
} int find(int x) {
return x == fa[x] ? x : fa[x] = find(fa[x]);
} bool check(int u,int v) {
int u1 = find(u), u2 = find(u + Mx);
int v1 = find(v), v2 = find(v + Mx);
if (u1 == v1) return ;
fa[u1] = v2; fa[v1] = u2;
return ;
} void solve2() {
for (int i = ; i <= (Mx << ); ++i) fa[i] = i;
for (int i = ; i * i <= (Mx << ); ++i) issqr[i * i] = ;
for (int i = n, j = n; i; ) {
for (bool flag = ; j; --j) {
if (!vis[a[j]]) { // 未出现过
for (int k = , tmp; ; ++k) {
tmp = k * k - a[j];
if (tmp <= ) continue; if (tmp > Mx) break;
if (vis[tmp] && (Ban[a[j]] || Ban[k * k - a[j]] || check(k * k - a[j], a[j]))) {
flag = ; fa[a[j]] = a[j], fa[a[j] + Mx] = a[j] + Mx; break;
}
}
if (!flag) break;
vis[a[j]] = true;
}
// 出现过:只有2*a=x^2(2,8...)的情况要特判,其他的都可以分到一集合里。
// 这些数最多出现两个,而且没有第三个敌人
else if (issqr[a[j] + a[j]]) {
if (Ban[a[j]]) break;
for (int k = , tmp; ; ++k) { // 判断是否有第三个敌人
tmp = k * k - a[j];
if (tmp < ) continue; if (tmp > Mx) break;
if (vis[tmp] && k * k != a[j] * ) { flag = ; break; }
}
if (!flag) break;
Ban[a[j]] = true; // a[j]以前一个,现在一个 a[j]以后不能再有了。
}
}
if (!j) break;
ans.push_back(j);
for (; i > j; --i) fa[a[i]] = a[i], fa[a[i] + Mx] = a[i] + Mx, vis[a[i]] = Ban[a[i]] = ;
}
} int main() {
n = read();int K = read();
for (int i=; i<=n; ++i) a[i] = read(), Mx = max(Mx, a[i]);
if (K == ) solve1();
else solve2();
printf("%d\n",ans.size() + );
for (int i=ans.size()-; i>=; --i) printf("%d ",ans[i]); putchar('\n');
return ;
}
P3940 分组的更多相关文章
- 2019.8.3 NOIP模拟测试12 反思总结【P3938 斐波那契,P3939 数颜色,P3940 分组】
[题解在下面] 早上5:50,Gekoo同学来到机房并表态:“打暴力,打暴力就对了,打出来我就赢了.” 我:深以为然. (这是个伏笔) 据说hzoi的人还差两次考试[现在是一次了]就要重新分配机房,不 ...
- [洛谷P3940]:分组(贪心+并查集)
题目传送门 题目描述 小$C$在了解了她所需要的信息之后,让兔子们调整到了恰当的位置.小$C$准备给兔子们分成若干个小组来喂恰当的胡萝卜给兔子们吃.此时,$n$只兔子按一定顺序排成一排,第$i$只兔子 ...
- 题解 P3940 分组
有些梦想虽然遥不可及,但不是不可能实现.只要我足够的强. 前言 调了挺长时间的,并查集合并的时候需要 find 一下,不然会炸内存.... 解题思路 参考了题解区一篇思路非常好的题解,在这里讲一下自己 ...
- 一条Sql语句分组排序并且限制显示的数据条数
如果我想得到这样一个结果集:分组排序,并且每组限定记录集的数量,用一条SQL语句能办到吗? 比如说,我想找出学生期末考试中,每科的前3名,并按成绩排序,只用一条SQL语句,该怎么写? 表[TScore ...
- xamarin android ListView手动分组
xamarin的listview控件其实自带有分组方法,关于xamarin listview的自带分组方法请自行参考官方文档,我这里只写自己写的分组方法.xamarin自带的分组好是好,功能多,但是加 ...
- [Java Collection]List分组之简单应用.
前言 今天有一个新需求, 是对一个List进行分组, 于是便百度到一些可用的代码以及我们项目使用的一些tools, 在这里总结下方便以后查阅. 一: 需求 现在我们一个数据库表t_series_val ...
- TSQL 分组集(Grouping Sets)
分组集(Grouping Sets)是多个分组的并集,用于在一个查询中,按照不同的分组列对集合进行聚合运算,等价于对单个分组使用“union all”,计算多个结果集的并集.使用分组集的聚合查询,返回 ...
- SQL Server 动态行转列(参数化表名、分组列、行转列字段、字段值)
一.本文所涉及的内容(Contents) 本文所涉及的内容(Contents) 背景(Contexts) 实现代码(SQL Codes) 方法一:使用拼接SQL,静态列字段: 方法二:使用拼接SQL, ...
- SolrNet高级用法(分页、Facet查询、任意分组)
前言 如果你在系统中用到了Solr的话,那么肯定会碰到从Solr中反推数据的需求,基于数据库数据生产索引后,那么Solr索引的数据相对准确,在电商需求中经常会碰到菜单.导航分类(比如电脑.PC的话会有 ...
随机推荐
- UVA-1152-4 Values whose Sum is 0---中途相遇法
题目链接: https://cn.vjudge.net/problem/UVA-1152 题目大意: 给出4个数组,每个数组有n个数,问有多少种方案在每个数组中选一个数,使得四个数相加为0. n &l ...
- struts2(2.0.x到2.1.2版本)的核心和工作原理(转)
在学习struts2之前,首先我们要明白使用struts2的目的是什么?它能给我们带来什么样的好处? 设计目标 Struts设计的第一目标就是使MVC模式应用于web程序设计.在这儿MVC模式的好处就 ...
- sprintf格式化字符串安全问题
先看sprintf用法: 定义和用法 sprintf() 函数把格式化的字符串写入变量中. arg1.arg2.++ 参数将被插入到主字符串中的百分号(%)符号处.该函数是逐步执行的.在第一个 % 符 ...
- 对json的理解?
回答一: a.JSON对象:以 ”{“ 开始,以 ”}” 结束,里面则是一系列的键(key)值(value)对,键和值用 ”:” 分开,每对键值对之间用 ”,” 分开.参考以下语法: {key1:va ...
- jq页面加载问题
Window.onload=function(){ //页面加载,不能同时编写多个,最后面的会覆盖前面的 } $(document).ready(function(){ //页面加载,能同时编写多 ...
- robotframwork接口测试(四)—其他库的安装
怎么知道自己的RF已经有哪些库了,可以看python安装目录下Python27\Lib\site-packages这个文件夹,有的话就可以直接引入了. 没有的话,就安装了. 1. 命令安装:这种最方便 ...
- 非接触式读卡器13.56MHZ芯片:SI522
对于现在的智能锁市场需求竞争极大,中小型厂家月销量更是在慢慢增长.刷卡功能更是智能锁的标配功能,当然13.56Mhz芯片现在讲究的就是超低功耗,为满足市场需求专注于物联网多年的中科微强力推出了13.5 ...
- MySQL进阶(视图)---py全栈
目录 mysql进阶(视图)---py全栈 一.什么是视图? 二.视图的特性 三.视图的优点 四.使用场合 五.视图基本操作 六.案例 mysql进阶(视图)---py全栈 一.什么是视图? 视图是从 ...
- echarts中跨域动态获取数据时,当某些对应的数据为空时,鼠标滑动到所在位置卡死
才疏学浅,万望指点. formatter: function (params) { var rel = params[0].name + "<br />"; rel + ...
- HAProxy负载均衡策略
HAProxy是一个使用C语言编写的自由及开放源代码软件,其提供高可用性.负载均衡,以及基于TCP和HTTP的应用程序代理.HAProxy是支持虚拟主机的,HAProxy的优点能够补充Nginx的一些 ...