POJ3241 最小曼哈顿距离生成树 - 真有趣哇
(有任何问题欢迎留言或私聊 && 欢迎交流讨论哦
Catalog
Problem:Portal传送门
原题目描述在最下面。
给你n个坐标,求最小曼哈顿距离生成树。
Solution:
请一定要理解:
有一个剪枝:把坐标分成\(8\)块,在一个\(45\)度区间内,只需要向与之距离最近的点连边。
虽然一共有\(8\)个相对区域,但我们只需考虑\(4\)个,中心对称的不需要再连一次边。这\(4\)个区域坐标转化一下即可求解。
先考虑每个点\(y\)轴向右的\(45\)度区域,例如,对于\(A\)点\((x0,y0)\),\(B(x1,y1)\)在\(A\)点的那部分区域内。有\(x0\leq x1,\; y0-x0\leq y1-x1\)。而\(A\)只要向满足此条件的大于\(y0-x0\)的最小\(x+y\)点连边。
先把所有点按\(x,y\)坐标排序,从最后一个点往前处理(这样保证了\(x1>x0\)的问题),然后用树状数组维护最小的\(x+y\)。
再来思考坐标转化的问题,把[45,90]标记为1,[0,45]标记为2,[-45,0]标记为3,[-90,-45]标记为4.
从区域1到2,3到4只需要交换x,y坐标即可;从区域2到3只需要把x坐标去相反数即可。
AC_Code:
#include<bits/stdc++.h>
#define lson rt<<1
#define rson rt<<1|1
#define all(x) (x).begin(),(x).end()
#define mme(a,b) memset((a),(b),sizeof((a)))
#define fuck(x) cout<<"* "<<x<<"\n"
#define iis std::ios::sync_with_stdio(false)
using namespace std;
typedef long long LL;
typedef pair<int,int> pii;
const int MXN = 1e5 + 7;
const int MXE = 1e6 + 7;
const int INF = 0x3f3f3f3f;
int n, k, tot;
int ar[MXN], fa[MXN];
struct lp{
int x, y, id;
}cw[MXN], edge[MXE];
bool cmp(const lp &a, const lp &b){
if(a.x != b.x)return a.x < b.x;
return a.y < b.y;
}
bool cmp1(const lp &a, const lp &b){
return a.id < b.id;
}
//树状数组部分
struct BIT{
int w, p;
}bit[MXN];
int lowbit(int x){
return x&(-x);
}
void add(int x, int w, int p){
for(; x > 0; x -= lowbit(x)){
if(bit[x].w > w)bit[x].w = w, bit[x].p = p;
}
}
int query(int x){
int mmax = INF, p = -1;
for(; x <= n; x += lowbit(x)){
if(bit[x].w < mmax)mmax = bit[x].w, p = bit[x].p;
}
return p;
}
int Fi(int x){
return fa[x] == x? x: fa[x] = Fi(fa[x]);
}
void add_edge(int u,int v,int w){
edge[++tot].x = u;edge[tot].y = v;edge[tot].id = w;
}
int abd(int x){return (x < 0)? -x : x;}
int dist(int i, int j){
return abs(cw[i].x-cw[j].x)+abs(cw[i].y-cw[j].y);
}
void kruskal(){
sort(edge, edge + tot + 1, cmp1);
int cnt = n - k, ans;
for(int i = 0; i <= n; ++i)fa[i] = i;
for(int i = 0; i <= tot && cnt; ++i){
int pa = Fi(edge[i].x), pb = Fi(edge[i].y);
if(pa == pb)continue;
--cnt;
ans = edge[i].id;
fa[pb] = pa;
}
printf("%d\n", ans);
}
/*
我们只需考虑在一块区域内的点,其他区域内的点可以通过坐标变换“移动”到这个区域内。为了方
便处理,我们考虑在y轴向右45度的区域。在某个点A(x0,y0)的这个区域内的点B(x1,y1)满足
x1≥x0且y1-x1>y0-x0。这里对于边界我们只取一边,但是操作中两边都取也无所谓。那么
|AB|=y1-y0+x1-x0=(x1+y1)-(x0+y0)。在A的区域内距离A最近的点也即满足条件的点中
x+y最小的点。因此我们可以将所有点按x坐标排序,再按y-x离散,用线段树或者树状数组维护
大于当前点的y-x的(最小的x+y)对应的点。时间复杂度O(NlogN)。
*/
int main(int argc, char const *argv[]){
#ifndef ONLINE_JUDGE
freopen("E://ADpan//in.in", "r", stdin);
//freopen("E://ADpan//out.out", "w", stdout);
#endif
while(~scanf("%d%d", &n, &k)){
tot = -1;
for(int i = 0 ; i < n; ++i){
scanf("%d%d", &cw[i].x, &cw[i].y);
cw[i].id = i;
}
for(int dir = 1; dir <= 4; ++dir){
//坐标转换
if(dir % 2 == 0){
for(int i = 0; i < n; ++i)swap(cw[i].x, cw[i].y);
}else if(dir == 3){
for(int i = 0; i < n; ++i)cw[i].x = -cw[i].x;
}
//先x再y排序
sort(cw, cw +n, cmp);
//Discretize
for(int i = 0; i <= n; ++i){
ar[i] = cw[i].y - cw[i].x;
bit[i].w = INF; bit[i].p = -1;
}
sort(ar, ar + n);
int k = unique(ar, ar + n) - ar;
for(int i = n - 1; i >= 0; --i){
//按y-x编号
int p = lower_bound(ar, ar + k, cw[i].y - cw[i].x) - ar + 1;
int pos = query(p);//获取最小的y+x
if(pos != -1){
add_edge(cw[i].id, cw[pos].id, dist(i, pos));
}
//添加y+x
add(p, cw[i].y + cw[i].x, i);
}
}
kruskal();
}
return 0;
}
Problem Description:
POJ3241 最小曼哈顿距离生成树 - 真有趣哇的更多相关文章
- Hdu4311 || 4312Meeting point-1/-2 n个点中任意选一个点使得其余点到该点曼哈顿距离之和最小
Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submission( ...
- poj3241 曼哈顿最小距离生成树第k大的边
思路: 已知: 要生成曼哈顿距离最小生成树,一个点最多和四周8个点连线,那8个点分别是将那个点四周360度平分成8个区间,每个区间里面和那个点曼哈顿距离最小的点,所以如果有n个点,那么最多有4n条边, ...
- 某个点到其他点的曼哈顿距离之和最小(HDU4311)
Meeting point-1 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
- TZOJ 1689 Building A New Barn(求平面上有几个其它点求到n个点的曼哈顿距离最小)
描述 After scrimping and saving for years, Farmer John has decided to build a new barn. He wants the b ...
- HDU 4311 Meeting point-1(曼哈顿距离最小)
http://acm.hdu.edu.cn/showproblem.php?pid=4311 题意:在二维坐标中有n个点,现在要从这n个点中选出一个点,使得其他点到该点的曼哈顿距离总和最小. 思路: ...
- hdu 4311 & 4312 Meeting point 曼哈顿距离之和最小
hdu 4311 题意 平面上\(n(n\leq 1e5)\)个点,找一个点到其它所有点的曼哈顿距离之和最小. 思路 如果是找一个坐标使得所有点到其曼哈顿距离之和最小,那么将\(n\)个横坐标排个序, ...
- 15 Puzzle (4乘4谜题) IDA*(DFS策略与曼哈顿距离启发) 的C语言实现
大家好!这是我的第一篇博客,由于之前没有撰写博客的经验,并且也是初入计算机和人工智能领域,可能有些表述或者理解不当,还请大家多多指教. 一.撰写目的 由于这个学期在上算法与数据结构课程的时候,其中一个 ...
- 曼哈顿距离MST
https://www.cnblogs.com/xzxl/p/7237246.html 讲的不错 /* 曼哈顿距离最小生成树 poj 3241 Object Clustering 按照上面的假设我们先 ...
- Atitti knn实现的具体四个距离算法 欧氏距离、余弦距离、汉明距离、曼哈顿距离
Atitti knn实现的具体四个距离算法 欧氏距离.余弦距离.汉明距离.曼哈顿距离 1. Knn算法实质就是相似度的关系1 1.1. 文本相似度计算在信息检索.数据挖掘.机器翻译.文档复制检测等领 ...
随机推荐
- PHP FILTER_VALIDATE_BOOLEAN 过滤器
定义和用法 FILTER_VALIDATE_BOOLEAN 过滤器把值作为布尔选项来验证. Name: "boolean" ID-number: 258 可能的返回值: 如果是 & ...
- java——文件
- Minimum Snap轨迹规划详解(1)轨迹规划
一. 轨迹规划是什么? 在机器人导航过程中,如何控制机器人从A点移动到B点,通常称之为运动规划.运动规划一般又分为两步: 1.路径规划:在地图(栅格地图.四\八叉树.RRT地图等)中搜索一条从A点到B ...
- SQL中忘记用户登陆密码该如何修改
1.每个数据库登陆之前都必须先启动它本身的数据服务,SQL数据库也不例外,首先我们要做的是先打开我们的SQL数据服务! 2.随后在我们的开始菜单中找到我们的SQL启动图标,打开即可 3.弹出登录窗体( ...
- (转)XMPP协议原理
本文介绍XMPP协议原理及相关信息. XMPP协议简介 XMPP(Extensible Messageing and Presence Protocol:可扩展消息与存在协议)是目前主流的四种IM ...
- LInux文件基础知识和文件目录操作(二)文件I/O操作
1.文件I/O操作分为两部分来讲解: 第一部分是非缓冲文件操作,这种操作适合于比较小规模文件的读写和对实时性要求很高的设备的数据通信,这类操作是系统调用提供的: 第二部分是缓冲文件操作,所面向的则是大 ...
- RFS自动化测试工具安装与使用总结
转载:http://blog.csdn.net/a5650892/article/details/77826021 一,调试1,在调试时,总时提示“无法打开浏览器”解决办法:1,把浏览器的代理关闭2, ...
- robotframework+selenium2library之上传本地文件
针对将本地的文件上传到测试系统,selenium2library提供了一个关键词 choose file choose file jquery=*[name='Filedata']+label: ...
- go gin
1.安装 go get -u github.com/gin-gonic/gin 2. package main import "github.com/gin-gonic/gin" ...
- Java异常关闭资源的两种方式
try-catch-finally 常用,在异常关闭时应判断流是否为空 public class CloseableUtils { public static void closeable(Close ...