汕头市队赛 SRM1X T2 ——扫描线

绵津见-终 SRM 13

背景

“西瓜也是可以种在海上的!”——绵津见

然而种在海上的西瓜最需要防范的,是时不时会涌向瓜田的阵阵海浪。

幸好,身为海神的绵津见可以释放魔法“水平如镜”来阻止海浪拍打西瓜。

然而,当西瓜一个接一个成熟之时,它们就要离开瓜田,飘向遥远的彼岸。绵津见的魔法无法保护离开瓜田的西瓜们,但至少,也得知道西瓜们遭遇了多大的风浪啊。

描述

我们用一个坐标系来描述大海,绵津见的瓜田位于x轴下方,每当有一个西瓜成熟时,它会从x轴上一点出发,沿一条平行y轴的直线往y轴正方向前进。

某个时刻,海上会有一个海浪生成。每个海浪都是一条平行于x轴的线段,并且会往y轴负方向前进。当它达到x轴下方时,它会受到“水平如镜”的影响而消失。

所有西瓜、海浪每个时刻会前进一个单位距离,西瓜与西瓜之间,海浪与海浪之间互不影响,相互重叠的情况也是允许发生的。

当一个西瓜与海浪重叠时,认定为该西瓜遭遇该海浪的拍打,西瓜与海浪的运动不受此次拍打的影响,原路前进。

每个时刻的流程如下:

①在该时刻成熟的西瓜被摆放在x轴上,该时刻生成的海浪出现。

②进行一次判定操作:如果有西瓜和某海浪上的一点重合,判定为发生了一次拍打。

③所有西瓜前进一个单位距离,即y坐标加1。

④进行一次判定操作。

⑤所有海浪前进一个单位距离,即y坐标减1。y坐标为-1的海浪消失。

你需要回答每个西瓜、每个海浪共涉及多少次拍打。

输入格式

第一行两个整数n,m表示西瓜的数量和海浪的数量。

接下来n行,每行两个整数ti,xi描述一个西瓜,分别表示西瓜成熟的时间ti,成熟时出现在(xi,0)位置。

接下来m行,每行四个整数Ti,li,ri,yi描述一个海浪,分别表示海浪生成的时间ti,生成时两个端点为(li,yi),(ri,yi)。

输出格式

输出第一行n个数,第i个数表示第i个西瓜被拍打的次数。

输出第二行m个数,第i个数表示第i个海浪拍打到的西瓜数。

样例输入 样例输出
5 3
2 3
2 5
2 6
4 6
4 7
1 1 6 0
2 1 6 0
2 6 7 3
1 1 2 1 1
0 3 3

数据范围与约定

对于初:  ,,

对于续:  ,,

对于终:  ,,

样例解释

?不解释

————————————————————————————————————

这道题分析之后 我们发现西瓜和海浪相碰需要满足

设西瓜出发时间T1 位置x 海浪T2 l r 高度 y

l<=x<=r  T2-y<=T1<=T2+y

很明显这是符合扫描线的性质 所以我们可以将T排序 将 l x r 离散化一波

将询问拆为两个 相减得到答案

按T扫一波然后进行区间求和 因为点和矩阵相互影响是对称的

所以询问和求和是相反的 这样就可以直接维护一波答案了

代码简短但是略复杂 还是要花时间慢慢体会的

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int M=1e5+7;
int read(){
    int ans=0,f=1,c=getchar();
    while(c<‘0‘||c>‘9‘){if(c==‘-‘) f=-1; c=getchar();}
    while(c>=‘0‘&&c<=‘9‘){ans=ans*10+(c-‘0‘); c=getchar();}
    return ans*f;
}
int n,m;
int xs[M*3],xp;
int qp,ep;
int s1[3*M],s2[3*M],ans1[M],ans2[M];
int lowbit(int x){return x&-x;}
void add(int x,int v,int s[]){
    while(x<=xp){
        s[x]+=v;
        x+=lowbit(x);
    }
}
int query(int x,int s[]){
    int ans=0;
    while(x){ans+=s[x]; x-=lowbit(x);}
    return ans;
}
struct Q{
    int l,r,y,id,s;
    bool operator <(const Q& h)const{return y<h.y;}
    void cal(){
        ans1[id]+=(query(r,s1)-query(l-1,s1))*s;
        add(l,-s,s2);
        add(r+1,s,s2);
    }
}e[2*M];
struct pos{
    int x,T,id;
    bool operator <(const pos& h)const{return T<h.T;}
    void cal(){
        add(x,1,s1);
        ans2[id]=query(x,s2);
    }
}q[2*M];
void $(int&x){x=lower_bound(xs,xs+xp,x)-xs+1;}
int main()
{
    int l,r,k,x;
    n=read(); m=read();
    for(int i=0;i<n;i++){
        k=read(); x=read();
        q[qp++]=(pos){xs[xp++]=x,k,i};
    }
    for(int i=0;i<m;i++){
        k=read(); l=read(); r=read(); x=read();
        e[ep++]=(Q){l,r,k-x-1,i,-1};
        e[ep++]=(Q){l,r,k+x,i,1};
        xs[xp++]=l;
        xs[xp++]=r;
    }
    sort(xs,xs+xp);
    for(int i=0;i<qp;++i)$(q[i].x);
    for(int i=0;i<ep;++i)$(e[i].l),$(e[i].r);
    sort(q,q+qp);
    sort(e,e+ep);
    for(int i=0,j=0;i<ep;i++){
        while(j<qp&&q[j].T<=e[i].y) q[j++].cal();
        e[i].cal();
    }
    for(int i=0;i<n;i++) printf("%d ",ans2[i]); printf("\n");
    for(int i=0;i<m;i++) printf("%d ",ans1[i]); printf("\n");
    return 0;
}

时间: 2024-10-05 13:14:12

汕头市队赛 SRM1X T2 ——扫描线的相关文章

汕头市队赛 SRM13 T2

这道题很容易想到是二分 但是因为可能会爆LL 所以要加一波特判 #include<cstdio> #include<cstring> #include<algorithm> #define LL long long using namespace std; const LL M=1e6+7,mx=1e18; LL read(){ LL ans=0,c=getchar(); while(c<'0'||c>'9') c=getchar(); while(c>

汕头市队赛 SRM16 T2

描述 猫和老鼠,看过吧?猫来了,老鼠要躲进洞里.在一条数轴上,一共有n个洞,位置分别在xi,能容纳vi只老鼠.一共有m只老鼠位置分别在Xi,要躲进洞里,问所有老鼠跑进洞里的距离总和最小是多少. 输入格式 两个用空格隔开的整数m和n. 这一行m个数字分别表示老鼠的位置 接下来n行每行两个数字分别表示洞的位置和容纳量 输出格式 一个整数,表示最小的距离总和.(如果无解,输出-1) 样例输入 4 5 6 2 8 9 3 6 2 1 3 6 4 7 4 7 样例输出 11----------------

汕头市队赛 SRM14 T2 最长上升子序列

最长上升子序列 (tree.pas/c/cpp) 128MB 1s 有一个长度为n的序列a[i],其中1到n的整数各自在a[i]中出现恰好一次. 现在已知另一个等长的序列f[i],表示a[i]中以第i个位置结尾的最长上升子序列的长度,请还原出a[i]. 输入格式 第一行一个正整数n. 接下来一行n个数,其中第i个数表示f[i]. 输出格式 一行,n个整数,表示序列a[i],如果答案不唯一,任意输出一种. 样例输入 7 1 2 3 2 4 4 3 样例输出 1 4 5 2 7 6 3 样例解释 以

汕头市队赛 C KMP codeforces B. Image Preview

汕头市队赛题目传送门 codeforces题目传送门 这道题我的做法是 尝试先往左走然后往右走 或者先往右走然后往左走 然后注意一下枚举顺序就okay啦 #include<cstdio> #include<cstring> #include<algorithm> #define LL long long using namespace std; const int M=1e6+7; LL read(){ LL ans=0,f=1,c=getchar(); while(c

汕头市队赛SRM15

T1--czl SRM 15 众所周知,czl家养了一只可♂爱的***(已屏蔽),那只东西很贪吃,所以czl家很多零食仓库,然而这些仓库里有很多老鼠. 为了心爱的***,czl决定点燃纯艾条,用烟熏老鼠. 共有N个仓库,编号1-N. 假设陵陵在第i个仓库点燃艾条,烟雾就会充满该仓库,并向左右扩散Ai 的距离,接着所有|i-j|<=Ai 的仓库 j 的老鼠被消灭. 陵陵是个爱护环境的人,他想知道最少需要多少支艾条,才可以消灭所有老鼠. [输入格式] 第一行:一个正整数,代表 N. 第二行:N 个非

汕头市队赛 SRM13 T3

这道题可以贪心 维护一个答案队列 枚举位置 每次将比当前位置大的队尾全部替代掉 记录删了多少了就好了 #include<cstdio> #include<iostream> #include<cstring> #include<algorithm> #include<string> #define LL long long using namespace std; const int M=1e7+7; int n,k,cnt; char s[M]

汕头市队赛 SRM 09 A 撕书

A 撕书I-3 SRM 09 背景&&描述 琉璃在撕书.     书总共有n页,都悬浮在数轴上,第i页的位置为,上面写着一个数字.     琉璃从右往左撕书.假如看到了第i页,就把在第i页左边,且与之距离<=的书都撕掉.(第i页本身不撕)     夜子为了尽量地保全魔法书,决定偷偷在琉璃开始撕之前,增加一页.增加的这一页必须在所有书页的右边,数字随意.     夜子想知道,最少会有多少页书被撕毁. 输入格式 第一行一个整数n,表示书页数. 接下来n行,第i行的俩整数分别为和. 输出格

汕头市队赛 SRM 09 B 撕书

B 撕书II-3 SRM 09 背景&&描述 琉璃手头有一黑一白两本魔法书,一本是<缟玛瑙的不在证明>,另一本是<白色相簿1.5>     传说同时打开这两本书会有奇怪的事情发生.     琉璃打开一看,果然非常奇怪:两本书上都各自写着一个正整数(可能他买到盗版了),分别是a和b.     试图撕书的汀想借过来看看,但琉璃只告诉了他这俩数加起来的值x和异或起来的值y.     汀发现有很多种(a,b)满足琉璃告诉他的信息...你能帮他算出来有多少种吗? 输入格式 两

汕头市队赛 SRM 06 C 秀恩爱

C 秀恩爱 SRM 06 背景&&描述 KPM坐在直升机上俯瞰小渔村景象. 渔村可看作二维平面,密密麻麻地到处都是单身狗,KPM当前所在坐标为(sx,sy). KPM的后宫团们自发地聚集在一起为他送行,从空中看,后宫团形成了一个多边形. 当然了KPM是不在那个多边形内的. 直升机突然开始原地转圈,后宫团们因为想看着KPM的正脸,所以也跟着以KPM所在坐标为中心旋转. 后宫团所经之处单身狗尸横遍野.赶来救治伤员的医护人员想知道,多边形扫过的面积是多少. 注意,本题不保证横坐标互不相同.纵坐标