USACO 2020 OPEN Silver Problem 3. The Moo Particle
题意:
解法:
首先给出在本题中连通和连通块的定义:
连通:
两个粒子a,b连通,当且仅当ax≤bx、ay≤by或者bx≤ax、by≤ay。
如图,A,B两粒子是连通的,而C、D不是。
可以看出,本题中连通的定义类似于无向边。
连通块:
一个有n个粒子的粒子集合S被称为连通块,当且仅当该集合内的粒子可以通过相互作用仅留下任意一个粒子。
左侧的A图中的四粒子属于同一连通块,而右侧的B图中四粒子分别属于两个连通块。
50分暴力:
显然,最少留下的粒子(以下简称为点)数等于连通块的总数,故本题可以转化为求最小的连通块划分数。通过该结论可以写出50分暴力(n≤1000)。
#include <bits/stdc++.h>
using namespace std;
const int N = 1e5 + ;
int n, ans, fa[N], dx[N], dy[N];
bool cont (int x, int y) {
if (dx[x] <= dx[y] && dy[x] <= dy[y])
return true;
if (dx[x] >= dx[y] && dy[x] >= dy[y])
return true;
return false;
}
int find (int x) {
if (x != fa[x])
return fa[x] = find (fa[x]);
return x;
}
void merges (int x, int y) {
int xx = find (x), yy = find (y);
if (xx == yy)
return ;
fa[xx] = yy;
return ;
}
int main () {
freopen ("moop.in", "r", stdin);
freopen ("moop.out", "w", stdout);
scanf ("%d", &n);
for (int i = ; i <= n; i ++) {
scanf ("%d %d", &dx[i], &dy[i]);
fa[i] = i;
}
for (int i = ; i <= n; i ++)
for (int j = i + ; j <= n; j ++)
if (cont (i, j))
merges (i, j);
for (int i = ; i <= n; i ++)
if (fa[i] == i)
ans ++;
printf ("%d\n", ans);
return ;
}
满分解法:
O(n2)暴力连边判断连通块的方法显然是不行的。所以需要找到一种快速划分连通块的方法。
考虑按每个点的x坐标排序并记录排序完成后的y坐标并用数组(a)记录,则可以快速离散化所有点的坐标。(当两点x相同时两点必定连通,故不需要特别讨论x坐标相同的情况)
考虑一个在x轴上连续的点集,满足该点集最左侧的点的y坐标比最右侧的点的y坐标小。(如图)
L、R分别为该点集中最左侧、最右侧的点。
显然,L与R连通。
将点集中剩下的点分为三类:
A类:在R点左上方(如A点)
B类:在R点左下方并且在L点右上方(如B点)
C类:在L点右下方(如C点)
对每类点的连通情况进行讨论:
A类:
因为A点在R点左上方,故A点y坐标比L大、x坐标比L大。所以该点与L点连通。
B类:
因为B点在L点右上方和R点左下方,故与L、R点都连通。
C类:
因为C点在L点右下方,故C点y坐标比R小,x坐标比R小,所以该店与R点连通。
综上所述,A、B、C类点都与L、R中的一点连通,而L又与R点连通。
可得到结论:任意一个满足最左侧的点的y坐标比最右侧的点的y坐标小的在x轴上连续的点集都满足连通块的定义。
由此可以得到一种O(n)的贪心的连通块划分方法:从左往右扫描所有点,若当前点的y值小于上一个点所属的连通块的L点的y值,则从该点开始新建一个连通块,并记录该点的y坐标。
但此贪心方法有明显错误,如图:
按照这种贪心方法,该点集会被划分为(A、B、C)和(D、E)两个连通块。实际上(A、B、C、D、E)即为一个连通块。
此时可以发现,若记录划分到当前点前每个连通块的L点的y坐标,并在每个点用二分算法判断该点y坐标最大大于哪个L点(设为l点)y坐标并将l点之后的所有点与l点合并为一个连通块,就可以在logn时间内更加准确地判断出到目前点为止最小的连通块划分数。
但这种判断方法仍然有问题,如图:
点(A、B、C、D、E、F)为一个连通块,但按照当前贪心方法会划分出(A、B、C、D、E)和(F)两连通块。故我们需要进一步优化贪心方法。
在原贪心方法中,我们记录的是到目前点为止,划分出的所有连通块的L节点的y值,但并入一个连通块并不一定需要大于该连通块的L点,仅需要在该联通块最右侧的点的右侧并大于该连通块的最小点即可。
故在合并连通块的时候,可以把原连通块的L节点的y值更新为现连通块L节点的y值。
这样即可求出最小连通块划分数。
代码如下:
#include <bits/stdc++.h>
using namespace std;
const int N = 1e5 + ;
int n, a[N], cnt, mn[N];
pair <int, int> p[N];
int main () {
freopen ("moop.in", "r", stdin);
freopen ("moop.out", "w", stdout);
scanf ("%d", &n);
for (int i = ; i <= n; i ++)
scanf ("%d %d", &p[i].first, &p[i].second);
sort (p + , p + n + );
for (int i = ; i <= n; i ++)
a[i] = p[i].second;
for (int i = ; i <= n; i ++) {
if (i == || -mn[cnt] > a[i]) {
cnt ++;
mn[cnt] = -a[i];
continue ;
}
if (cnt == )
continue;
if (-mn[cnt - ] <= a[i]) {
int mnn = mn[cnt];
int ls = lower_bound (mn + , mn + cnt + , -a[i]) - mn;
cnt = ls;
mn[cnt] = mnn;
}
}
printf ("%d\n", cnt);
return ;
}
USACO 2020 OPEN Silver Problem 3. The Moo Particle的更多相关文章
- USACO翻译:USACO 2013 NOV Silver三题
USACO 2013 NOV SILVER 一.题目概览 中文题目名称 未有的奶牛 拥挤的奶牛 弹簧牛 英文题目名称 nocow crowded pogocow 可执行文件名 nocow crowde ...
- USACO翻译:USACO 2013 DEC Silver三题
USACO 2013 DEC SILVER 一.题目概览 中文题目名称 挤奶调度 农场航线 贝西洗牌 英文题目名称 msched vacation shuffle 可执行文件名 msched vaca ...
- USACO翻译:USACO 2014 DEC Silver三题
USACO 2014 DEC SILVER 一.题目概览 中文题目名称 回程 马拉松 奶牛慢跑 英文题目名称 piggyback marathon cowjog 可执行文件名 piggyback ma ...
- USACO翻译:USACO 2012 FEB Silver三题
USACO 2012 FEB SILVER 一.题目概览 中文题目名称 矩形草地 奶牛IDs 搬家 英文题目名称 planting cowids relocate 可执行文件名 planting co ...
- USACO翻译:USACO 2014 FEB SILVER 三题
USACO 2014 FEB SILVER 一.题目概览 中文题目名称 自动打字 路障 神秘代码 英文题目名称 auto rblock scode 可执行文件名 auto rblock scode 输 ...
- USACO 2013 Nov Silver Pogo-Cow
最近因为闲的蛋疼(停课了),所以开始做一些 USACO 的银组题.被完虐啊 TAT 貌似 Pogo-Cow 这题是 2013 Nov Silver 唯一一道可说的题目? Pogo-Cow Descri ...
- USACO翻译:USACO 2014 MARCH Silver三题
USACO 2014 MARCH 一.题目概览 中文题目名称 农田灌溉 懒牛 牛叫 英文题目名称 irrigation lazy mooomoo 可执行文件名 irrigation lazy mooo ...
- [9018_1592]USACO 2014 Open Silver Fairphoto
题目描述 Farmer John's N cows (1 <= N <= 100,000) are standing at various positions along a long o ...
- [USACO 2012 Mar Silver] Landscaping【Edit Distance】
传送门:http://www.usaco.org/index.php?page=viewproblem2&cpid=126 好题啊好题,一开始就输给了这道题的想法! 先把原始状态以及目标状态换 ...
随机推荐
- Java实现蓝桥杯VIP算法训练 二元函数
试题 算法训练 二元函数 资源限制 时间限制:1.0s 内存限制:256.0MB 问题描述 令二元函数f(x,y)=ax+by,a和b为整数,求一个表达式S的值. 只有满足以下要求的表达式才是合法的: ...
- Java实现 蓝桥杯VIP 算法提高 扫雷
算法提高 扫雷 时间限制:1.0s 内存限制:256.0MB 问题描述 扫雷游戏你一定玩过吧!现在给你若干个n×m的地雷阵,请你计算出每个矩阵中每个单元格相邻单元格内地雷的个数,每个单元格最多有8个相 ...
- Java实现 洛谷 P1598 垂直柱状图
题目描述 写一个程序从输入文件中去读取四行大写字母(全都是大写的,每行不超过100个字符),然后用柱状图输出每个字符在输入文件中出现的次数.严格地按照输出样例来安排你的输出格式. 输入格式 四行字符, ...
- uniapp每隔几秒执行一下网络请求(h5端亲测可以,其他端未测试)
methods: { //执行网络请求 run() { uni.request({ method: 'GET',//请求方式 url: ‘’//请求地址 }).then(res=>{ conso ...
- (数据科学学习手札86)全平台支持的pandas运算加速神器
本文示例代码已上传至我的Github仓库https://github.com/CNFeffery/DataScienceStudyNotes 1 简介 随着其功能的不断优化与扩充,pandas已然成为 ...
- python—socket编程
一:客户端/服务器 架构 1.硬件C/S架构:(例如,打印机) 2.软件C/S架构:互联网中处处是C/S架构 腾讯作为服务端为你提供视频,你得下个腾讯视频客户端才能看它的视频 C/S架构与socket ...
- python—列表,元组,字典
——列表:(中括号括起来:逗号分隔每个元素:列表中的元素可以是数字,字符串,列表,布尔值等等) (列表元素可以被修改) list(类) (有序的) [1]索引取值:切片取值:for循环:whi ...
- redis的5种数据结构和基本操作
1.字符串(string) 1.1设置值 set key value [ex seconds] [px milliseconds] [nx|xx] 例如: 127.0.0.1:6379> set ...
- spring cloud 集成分布式配置中心 apollo(单机部署apollo)
一.什么是apollo? Apollo(阿波罗)是携程框架部门研发的分布式配置中心,能够集中化管理应用不同环境.不同集群的配置,配置修改后能够实时推送到应用端,并且具备规范的权限.流程治理等特性,适用 ...
- EIGRP-15-其他和高级的EIGRP特性-1-路由器ID
与很多协议一样, EIGRP也使用了路由器ID (RTD)的概念,用一个4字节的编号来标识某个路由器实例.每个地址家族实例拥有自已独立的RID.工程师可以在一台路由器上,为多个EIGRP进程和地址家族 ...