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的话会有 ...
随机推荐
- JavaScript设计模式之设计原则
何为设计 即按照哪一种思路或者标准来实现功能,功能相同,可以有不同的设计方案来实现 伴随着需求的增加,设计的作用就会体现出来,一般的APP每天都在变化,更新很快,需求不断在增加,如果设计的不好,后面很 ...
- node.js的npm命令常见错误及解决方案
使用npm命令进行模块安装的时候场出现各种错误,本文总结我所遇到的各种错误,并提供解决方案.(大部分内容为网上收集) 首先使用淘宝 NPM 镜像 大家都知道国内直接使用 npm 的官方镜像是非常慢的, ...
- x-frame-options、iframe与iframe的一些操作
iframe的子操作父窗口,父操作子窗口: test.php: <!DOCTYPE html> <html> <head> <title>test< ...
- 6、Web Service-拦截器
1.为什么CXF设置拦截器 为了在webservice请求过程中,能动态操作请求和响应数据, CXF设计了拦截器.拦截器分类 1.按所处的位置分:服务器端拦截器,客户端拦截器 2.按消息的方向分:入拦 ...
- MVC5新特性(一)之RouteAttribute打造自己的URL规则
1.RouteAttribute概述 RouteAttribute的命名空间是System.Web.Mvc,区别与web api的RouteAttribute(它的命名空间是System.Web.Ht ...
- 虚拟机和主机文件实时同步 -- winsshfs的快速入手
之前在公司使用mac ,并且通过mac下的osfuse和sshfs连接,直接将虚拟机的文件目录同步到了本地,并且可以进行实时操作修改,对于写项目,确实是省了很大一部分上传的精力. 于是在自己的win下 ...
- git地址
登录地址:https://git.oschina.net/signup API地址:http://git.oschina.net/progit/
- Java语言实现通过Ajax抓取后台数据及图片
1.Java语言实现通过Ajax抓取后台数据及图片信息 1.1数据库设计: create table picture( pic_id number not null, pic_name )not nu ...
- ASP.NET MVC 自动模型验证
经常看到这个代码 在controller 中写入验证模型,每个需要验证的action 都写-.. ,就问你烦不烦~ 可以利用 ASP.NET MVC 的 action 拦截机制 自动处理. 1 新建验 ...
- [iOS]app的生命周期
对于iOS应用程序,关键的是要知道你的应用程序是否正在前台或后台运行.由于系统资源在iOS设备上较为有限,一个应用程序必须在后台与前台有不同的行为.操作系统也会限制你的应用程序在后台的运行,以提高电池 ...