2018 ACM-ICPC 徐州赛区 网络赛G题

(有任何问题欢迎留言或私聊 && 欢迎交流讨论哦

Catalog

Problem:网络预赛-G Trace

Portal传送门
 原题目描述在最下面。
 我理解的题意大概是:有 n n n次涨潮和退潮,每次的范围是个 x × y x\times y x×y的矩形,求n次涨退潮后,潮水痕迹的长度。
 不存在此 i , j ∈ [ 1 , n ] , i ≠ j , x i ≤ x j 且 y i ≤ y j i,j\in[1,n], i\neq j,xi\leq xj且yi\leq yj i,j[1,n],i̸=j,xixjyiyj

Solution:

 每次潮水可能会冲刷掉之前的潮水的一部分痕迹,但是它又不会完全冲刷。
 考虑从最后一次潮水开始往前算贡献,分x,y方向计算。
 假设此次潮水的范围是 [ x i , y i ] [xi, yi] [xi,yi],找出大于等于 x i xi xi的最大 Y Y Y,则 y y y方向的新增痕迹长度为 y i − Y yi-Y yiY x x x方向同理。
 这样问题就转化为一个求区间最值的问题了,线段树秒。

AC_Code:
#include<bits/stdc++.h>
#define lson rt<<1
#define rson rt<<1|1
using namespace std;
typedef long long LL;
typedef unsigned long long uLL;
const int mod = 1e9+7;
const int MXN = 1e6 + 7;
int n;
int sumx[MXN<<2], sumy[MXN<<2];
int ar[MXN], br[MXN], le[MXN], ri[MXN];
void update(int op,int p,int c,int l,int r,int rt){
    if(l == r){
        if(op == 1) sumx[rt] = max(c,sumx[rt]);
        else sumy[rt] = max(c,sumy[rt]);
        return;
    }
    int mid = (l+r)>>1;
    if(p<=mid)update(op,p,c,l,mid,lson);
    else update(op,p,c,mid+1,r,rson);
    if(op == 1)sumx[rt] = max(sumx[lson], sumx[rson]);
    else sumy[rt] = max(sumy[lson], sumy[rson]);
}
int query(int op,int L,int R,int l,int r,int rt){
    if(L<=l&&r<=R){
        if(op == 1) return sumx[rt];
        else return sumy[rt];
    }
    int mid = (l+r)>>1;
    if(L>mid)return query(op,L,R,mid+1,r,rson);
    else if(R<=mid)return query(op,L,R,l,mid,lson);
    else {
        return max(query(op,L,mid,l,mid,lson),query(op,mid+1,R,mid+1,r,rson));
    }
}
int main(int argc, char const *argv[]){
    scanf("%d", &n);
    int k = 0;
    for(int i = 1; i <= n; ++i){
        scanf("%d%d", &le[i], &ri[i]);
        ar[k++] = le[i];
        ar[k++] = ri[i];
    }
    sort(ar, ar + k);
    k = unique(ar, ar + k) - ar;
    memset(sumx, 0, sizeof(sumx));
    memset(sumy, 0, sizeof(sumy));
    LL ans = 0;
    for(int i = n; i >= 1; --i){
        int a = le[i], b = ri[i];
        le[i] = lower_bound(ar, ar + k, le[i]) - ar + 1;
        ri[i] = lower_bound(ar, ar + k, ri[i]) - ar + 1;
        int my = query(1, le[i], k, 1, k, 1);//找出大于xi的最大Y
        ans += b - my;
        int mx = query(2, ri[i], k, 1, k, 1);//找出大于yi的最大X
        ans += a - mx;
        //printf("%d %d\n", mx, my);
        update(1, le[i], ar[ri[i]-1], 1, k, 1);//更新此xi下的yi
        update(2, ri[i], ar[le[i]-1], 1, k, 1);
    }
    printf("%lld\n", ans);
    return 0;
}

Problem Description:

这里写图片描述
这里写图片描述
这里写图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值