zoukankan      html  css  js  c++  java
  • 【转化,排序,二分】

    问题描述

    给定平面上n个点。

    求两条直线,这两条直线互相垂直,而且它们与x轴的夹角为45度,并且n个点中离这两条直线的曼哈顿距离的最大值最小。

    两点之间的曼哈顿距离定义为横坐标的差的绝对值与纵坐标的差的绝对值之和,一个点到两条直线的曼哈顿距离是指该点到两条直线上的所有点的曼哈顿距离中的最小值。

    输入格式

    第一行包含一个数n。

    接下来n行,每行包含两个整数,表示n个点的坐标(横纵坐标的绝对值小于109)。

    输出格式
    输出一个值,表示最小的最大曼哈顿距离的值,保留一位小数。
    样例输入
    4
    1 0
    0 1
    2 1
    1 2
    样例输出
    1.0
    数据规模与约定

    对于30%的数据,n<=100。

    对于另外30%的数据,坐标范的绝对值小于100。

    对于100%的数据,n<=10^5。

    -------------------------------------------------------------------------------------

    思路:由于两条直线和坐标轴夹角为45度,且相互垂直,不妨把坐标系旋转45度,使得直线和坐标轴平行。解决最大值最小这类问题一般用二分来做,二分答案,看能否构造出答案出来。问题转化为能否将所有点分成两个部分,使得其中一部分的x值之差恒小于等于d,另一部分y值之差恒小于等于d,如果能那么答案d/2就存在,否则答案不存在,据此缩小答案范围。在判断答案是否存在的时候,将点按x坐标排序,从小到大枚举x的最小值,那么x的最大值是单调递增的,那么剩下的点由前p和和后q个构成,它们的纵坐标的最大差可以一次性预处理,然后O(1)得到。总复杂度n(log n + log maxans)

     1 #include <iostream>
     2 #include <cstdio>
     3 #include <algorithm>
     4 #include <cstring>
     5 #include <map>
     6 #include <queue>
     7 #include <cmath>
     8 #include <vector>
     9 #include <ctime>
    10 #define mem0(a) memset(a, 0, sizeof(a))
    11 #define lson l, m, rt << 1
    12 #define rson m + 1, r, rt << 1 | 1
    13 #define define_m int m = (l + r) >> 1
    14 #define LL long long
    15 #define Rep(a, b) for(int a = 0; a < b; a++)
    16 #define lowbit(x) ((x) & (-(x)))
    17 const int dx[4] = {1, 0, -1, 0};
    18 const int dy[4] = {0, -1, 0, 1};
    19 const int INF = 1e9 + 7;
    20 const double eps = 1e-3;
    21 const double con = sqrt(2.0) / 2;
    22 const double maxd = 1e20;
    23 typedef double db;
    24 using namespace std;
    25 struct Node {
    26     double x, y;
    27     void inp() {
    28         int tx, ty;
    29         scanf("%d %d", &tx, &ty);
    30         x = (tx + ty) * con;
    31         y = (ty - tx) * con;
    32     }
    33     bool operator < (const Node &opt) const {
    34         return x < opt.x || x == opt.x && y < opt.y;
    35     }
    36 } node[100010];
    37 typedef double arr[100010];
    38 arr pre_min, pre_max, suf_min, suf_max;
    39 int n;
    40 void Init() {
    41     pre_min[0] = pre_max[0] = node[0].y;
    42     suf_min[n - 1] = suf_max[n - 1] = node[n - 1].y;
    43     for (int i = 1; i < n; i++) {
    44         pre_max[i] = max(pre_max[i - 1], node[i].y);
    45         pre_min[i] = min(pre_min[i - 1], node[i].y);
    46     }
    47     for(int i = n - 2; i >= 0; i--) {
    48         suf_max[i] = max(suf_max[i + 1], node[i].y);
    49         suf_min[i] = min(suf_min[i + 1], node[i].y);
    50     }
    51 }
    52 bool Check(double d) {
    53     int pos = 0;
    54     for (int i = 0; i < n; i++) {
    55         while (pos < n - 1 && node[pos + 1].x - node[i].x <= d) pos++;
    56         double tmp1 = 0, tmp2 = maxd;
    57         if (i) {
    58             tmp1 = pre_max[i - 1];
    59             tmp2 = pre_min[i - 1];
    60         }
    61         if (pos < n - 1) {
    62             tmp1 = max(tmp1, suf_max[pos + 1]);
    63             tmp2 = min(tmp2, suf_min[pos + 1]);
    64         }
    65         if (tmp1 - tmp2 <= d) return 1;
    66     }
    67     return 0;
    68 }
    69 int main() {
    70     //freopen("in.txt", "r", stdin);
    71     cin >> n;
    72     for (int i = 0; i < n; i++) {
    73         node[i].inp();
    74     }
    75     sort(node, node + n);
    76     Init();
    77     double l = 0, r = maxd;
    78     while (fabs(r - l) > eps) {
    79         double m = (l + r) / 2;
    80         if (Check(m)) r = m;
    81         else l = m;
    82     }
    83     printf("%.1f
    ", l * con);
    84     return 0;
    85 }
    View Code
  • 相关阅读:
    Python04 range()方法的使用、turtle.textinput()方法和write()的使用、turtle.numinput()的使用
    SpringBoot12 QueryDSL02之利用QueryDSL实现多表关联查询
    SpringBoot12 QueryDSL01之QueryDSL介绍、springBoot项目中集成QueryDSL、利用QueryDSL实现单表RUD、新增类初始化逻辑
    ES02 变量、数组、对象、方法
    JavaScript问题01 js代码放在header和body的区别
    Python03 字符串类型、强制类型转化、列表、元组、字典、集合
    Python02 标准输入输出、数据类型、变量、随记数的生成、turtle模块详解
    使用 Multipath TCP 为 iOS 创建备份连接(转)
    linux中的条件变量
    linux中网络编程<1>
  • 原文地址:https://www.cnblogs.com/jklongint/p/4352683.html
Copyright © 2011-2022 走看看