L3-009 长城

正如我们所知,中国古代长城的建造是为了抵御外敌入侵。在长城上,建造了许多烽火台。每个烽火台都监视着一个特定的地区范围。一旦某个地区有外敌入侵,值守在对应烽火台上的士兵就会将敌情通报给周围的烽火台,并迅速接力地传递到总部。

现在如图1所示,若水平为南北方向、垂直为海拔高度方向,假设长城就是依次相联的一系列线段,而且在此范围内的任一垂直线与这些线段有且仅有唯一的交点。

图 1

进一步地,假设烽火台只能建造在线段的端点处。我们认为烽火台本身是没有高度的,每个烽火台只负责向北方(图1中向左)瞭望,而且一旦有外敌入侵,只要敌人与烽火台之间未被山体遮挡,哨兵就会立即察觉。当然,按照这一军规,对于南侧的敌情各烽火台并不负责任。一旦哨兵发现敌情,他就会立即以狼烟或烽火的形式,向其南方的烽火台传递警报,直到位于最南侧的总部。

以图2中的长城为例,负责守卫的四个烽火台用蓝白圆点示意,最南侧的总部用红色圆点示意。如果红色星形标示的地方出现敌情,将被哨兵们发现并沿红色折线将警报传递到总部。当然,就这个例子而言只需两个烽火台的协作,但其他位置的敌情可能需要更多。

然而反过来,即便这里的4个烽火台全部参与,依然有不能覆盖的(黄色)区域。

图 2

另外,为避免歧义,我们在这里约定,与某个烽火台的视线刚好相切的区域都认为可以被该烽火台所监视。以图3中的长城为例,若A、B、C、D点均共线,且在D点设置一处烽火台,则A、B、C以及线段BC上的任何一点都在该烽火台的监视范围之内。

图 3

好了,倘若你是秦始皇的太尉,为不致出现更多孟姜女式的悲剧,如何在保证长城安全的前提下,使消耗的民力(建造的烽火台)最少呢?

输入格式:

输入在第一行给出一个正整数N(3 ≤ N ≤105),即刻画长城边缘的折线顶点(含起点和终点)数。随后N行,每行给出一个顶点的xy坐标,其间以空格分隔。注意顶点从南到北依次给出,第一个顶点为总部所在位置。坐标为区间[−109,109)内的整数,且没有重合点。

输出格式:

在一行中输出所需建造烽火台(不含总部)的最少数目。

这里最重要的就是理解非凸点要弹出,并且知道我们从南到北进行读取点,所以新加入的是北方的点

 

#include<bits/stdc++.h>
 
#define x first
#define y second

//#define MAXN 200011
using namespace std;
const int N = 5e5+ 10;
const int M = 2e4 + 10;
const int INF = 0x3f3f3f3f;
const int mod = 1e9 + 10;
typedef long long LL;
typedef pair<double, double>PDD;
typedef pair<int, int>PII;
typedef pair<int, string>PIS;
typedef pair<string, string>PSS;
int n, m, k, T;
int x[N],y[N];
set<int>se;
int stk[N];
int top;
bool check(int a,int b,int c)
{
    //这里除法==乘法,消去被除数 //向量ab在ac下面(kab<kac),b是凹点,这个时候b不是凸点(b在上面)
    return (LL)(x[c]-x[a])*(y[b]-y[a])<=(LL)(x[b]-x[a])*(y[c]-y[a]);//这里乘法可能要爆,毕竟1e9*1e9==1e18
}
int main()
{
    scanf("%d",&n);
    for(int i=0;i<n;++i)
    {
        scanf("%d %d",&x[i],&y[i]);
        if(top)
        {
            while(top>=2&&check(i,stk[top-1],stk[top-2]))top--;//b是凹点不要它了,stk[top-1]这个时候是栈顶 
            if(stk[top-1])se.insert(stk[top-1]);//set去重,可能多个这个点 //找到凸点了入栈
        }
        stk[top++]=i;//m 
    }
    cout<<se.size();
    return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值