IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> 第46屆ICPC 東亞洲區域賽(澳門)46届澳门站 C题 计算几何+双指针 -> 正文阅读

[C++知识库]第46屆ICPC 東亞洲區域賽(澳門)46届澳门站 C题 计算几何+双指针

[传送门]

题意

一直角坐标系中,BaoBao在原点位置,平面上有n个发射装置,任意两个发射器之间都有激光束,共(n-1)*n/2条激光,BaoBao的朋友可以去除任何发射器,问最少去除多少发射器,能使BaoBao不会被激光所伤还能从 ( 0 , 0 ) (0,0) (0,0)逃逸到 ( 1 0 1 0 1 0 1 0 10 , 1 0 1 0 1 0 1 0 10 ) (10^{10^{10^{10^{10}}}},10^{10^{10^{10^{10}}}}) (1010101010,1010101010)

tip.发射器不会重合,也不会存在激光束穿过 ( 0 , 0 ) (0,0) (0,0)

思路&解法

首先需要知道计算计算几何的基础知识:极角排序、向量叉积及其意义

可以发现,在枚举一条过原点的直线,直线一侧的发射器都移除时,BaoBao才可以逃逸。但是直线太多,我们怎么处理呢。

如果学过计算几何的旋转卡壳,都能比较快的想到双指针枚举直线

先将所有点对原点极角排序,但是对于angel相同的需要去重加权,然后双指针指向枚举的直线一侧的角度最小的点和角度最大的点,双指针不断滑动表示枚举直线,枚举同时维护ans。

p 0 p_0 p0?位原点,枚举 p l p_l pl?,寻找对应 p r p_r pr?,当 p 0 p l ? \vec{p_0p_l} p0?pl? ? p 0 p r ? \vec{p_0p_r} p0?pr? ?的叉积 p 0 p l ? \vec{p_0p_l} p0?pl? ? p 0 p r + 1 ? \vec{p_0p_{r+1}} p0?pr+1? ?的叉积 异号,表示 p l p_l pl? p r p_r pr?都是直线一侧的点。

tips.由于坐标达到 ∣ x ∣ , ∣ y ∣ ≤ 1 0 9 |x|,|y| \leq 10^9 x,y109,所以精度要求就是1e-18(小数点后18位)(大概是这样),所以储存angel时double的精度是不够的,要用到long double,但是计算过程中大量涉及叉积,坐标用long double计算速度就慢,会T,所以坐标的储存推荐使用long long(格点,且不推荐int,计算叉积过程中会爆int)

Accepted code

#include <bits/stdc++.h>
using namespace std;
#define db long double
#define maxn 1000005
#define ll long long
#define Vector point
#define id(x) ((x - 1) % (len - 1) + 1)
const db eps = 1e-18;

int _, n, len = 1;

struct point {
    ll x, y;
    db ag;
    point() {}
    point(ll xx, ll yy) : x(xx), y(yy) { ag = atan2((db)y, (db)x); }
    point operator-(const point &a) { return point(x - a.x, y - a.y); }
    bool operator<(const point &a) const {
        //         return atan2(y, x) < atan2(a.y, a.x);
        return ag < a.ag;
    }
};

point p[maxn], pp[maxn];
int num[maxn];

db Cross(Vector a, Vector b) { return a.x * b.y - a.y * b.x; }
db Dot(Vector a, Vector b) { return a.x * b.x + a.y * b.y; }

point p0(0, 0);
int ans;

bool sgn(db a, db b) { return fabs(a - b) < eps; }

bool sgn(point a, point b) {
    a = a - p0, b = b - p0;
    return fabs(a.ag - b.ag) < eps;
}

bool ck(int l, int r) {
    return Cross(pp[l] - p0, pp[r] - p0) * Cross(pp[l] - p0, pp[id(r + 1)] - p0) < eps 
    	   && !sgn(pp[l], pp[r]) && !sgn(pp[l], pp[id(r + 1)]);
}

int main() {
    for (scanf("%d", &_); _; _--) {
        len = 1;
        scanf("%d", &n);
        ans = n;
        for (int i = 0, x, y; i < n; i++) {
            scanf("%d%d", &x, &y);
            p[i] = point(x, y);
        }
        sort(p, p + n);
        // 去重加权
        for (int i = 0; i < n; i++) {
            if (i && sgn(p[i], p[i - 1])) {
                ++num[len - 1];
            } else {
                pp[len] = p[i];
                num[len++] = 1;
            }
        }
        if (len - 1 < 3) {
            printf("0\n");
            continue;
        }
        for (int i = 1; i < len; i++) num[i] += num[i - 1];
        ll r = 1, l = 1;
        for (; r < len; r++)
            if (ck(l, r) || id(r + 1) == id(l)) break;
        while (l < len) {
            for (;; ++r)
                if (ck(id(l), id(r)) || id(r + 1) == id(l)) break;
            if (id(l) <= id(r)) ans = min(ans, min(n - (num[id(r)] - num[id(l) - 1]), num[id(r)] - num[id(l) - 1]));
            else ans = min(ans, min(num[id(r)] + n - num[id(l) - 1], n - (num[id(r)] + n - num[id(l) - 1])));
            ++l;
        }
        printf("%d\n", ans);
    }
    return 0;
}

在这里插入图片描述

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-04-07 22:27:40  更:2022-04-07 22:28:13 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/10 20:28:01-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码