Vijos P1066 弱弱的战壕【多解,线段树,暴力,树状数组】

弱弱的战壕

描述

永恒和mx正在玩一个即时战略游戏,名字嘛~~~~~~恕本人记性不好,忘了-_-b。

mx在他的基地附近建立了n个战壕,每个战壕都是一个独立的作战单位,射程可以达到无限(“mx不赢定了?!?”永恒[email protected][email protected])。

但是,战壕有一个弱点,就是只能攻击它的左下方,说白了就是横纵坐标都不大于它的点(mx:“我的战壕为什么这么菜”ToT)。这样,永恒就可以从别的地方进攻摧毁战壕,从而消灭mx的部队。

战壕都有一个保护范围,同它的攻击范围一样,它可以保护处在它左下方的战壕。所有处于它保护范围的战壕都叫做它的保护对象。这样,永恒就必须找到mx的战壕中保护对象最多的点,从而优先消灭它。

现在,由于永恒没有时间来计算,所以拜托你来完成这个任务:

给出这n个战壕的坐标xi、yi,要你求出保护对象个数为0,1,2……n-1的战壕的个数。

格式

输入格式

第一行,一个正整数n(1<=n<=15000)
接下来n行,每行两个数xi,yi,代表第i个点的坐标
(1<=xi,yi<=32000)
注意:可能包含多重战壕的情况(即有数个点在同一坐标)

输出格式

输出n行,分别代表保护对象为0,1,2……n-1的战壕的个数。

样例1

样例输入1

5
1 1
5 1
7 1
3 3
5 5

样例输出1

1
2
1
1
0

限制

各点2s(算是宽限吧^_^)

来源

URAL1028战役版

题目链接:https://vijos.org/p/1066

分析:先给出我线段树写法吧,XX了几个小时,一行代码,三个错误,我也是无语了!好不容易过了样例,啪啪啪,打脸了,才过了三组数据,查了半天只要改了右区间就会过qwq!

此题是要我们求每次有一个新的点加入时,比它小的y有几个,因为,先对X从小到大排序以后,每次加入一个y,之前比它小的y保证x和y都小于它,个数就是这个星星的等级这就可以这样,一个1-32000的线段,每次加入一个y,加入了一个[y,32000]的线段,然后每次看比y少的点有几个,就是它当前被压在几条线段下!

下面给出线段树写法:

 1 #include <bits/stdc++.h>
 2 using namespace std;
 3 const int N=16666;
 4 int ans[N<<1];
 5 struct Node
 6 {
 7     int l,r,sum;
 8 }tree[N<<4];
 9 struct data
10 {
11     int x,y;
12 }f[N<<1];
13 inline bool cmp(data a,data b)
14 {
15     return (a.x==b.x)?(a.y<b.y):(a.x<b.x);
16 }
17 void buildtree(int l,int r,int pos)
18 {
19     tree[pos].sum=0;
20     tree[pos].l=l;
21     tree[pos].r=r;
22     if(l==r)
23         return;
24     int mid=(l+r)/2;
25     buildtree(l,mid,pos*2);
26     buildtree(mid+1,r,pos*2+1);
27 }
28 void update(int st,int ed,int pos)
29 {
30     if(tree[pos].l==st&&tree[pos].r==ed)
31     {
32         tree[pos].sum++;
33         return;
34     }
35     if(tree[pos].l==tree[pos].r)
36         return;
37     int mid=(tree[pos].l+tree[pos].r)/2;
38     if(mid>=ed)
39         update(st,ed,pos*2);
40     else if(st>mid)
41         update(st,ed,pos*2+1);
42     else
43     {
44         update(st,mid,pos*2);
45         update(mid+1,ed,pos*2+1);
46     }
47 }
48 int query(int c,int pos)
49 {
50     if(tree[pos].l==tree[pos].r)
51         return tree[pos].sum;
52     int mid=(tree[pos].l+tree[pos].r)/2;
53     if(c<=mid)
54         return tree[pos].sum+query(c,pos*2);
55     else return tree[pos].sum+query(c,pos*2+1);
56 }
57 int main()
58 {
59     int n;
60     cin>>n;
61     buildtree(0,32001,1);
62     for(int i=0;i<n;i++)
63         scanf("%d%d",&f[i].x,&f[i].y);
64     sort(f,f+n,cmp);
65     for(int i=0;i<n;i++)
66     {
67         ans[query(f[i].y,1)]++;
68         update(f[i].y,32001,1);
69     }
70     for(int i=0;i<n;i++)
71         cout<<ans[i]<<endl;
72     return 0;
73 }

此题数据太弱,所以用暴力同样可以解决问题!

 1 #include <bits/stdc++.h>
 2 using namespace std;
 3 const int N=16666;
 4 struct Node
 5 {
 6     int x,y;
 7 }p[N<<1];
 8 int n,ans[N<<1];
 9 int main()
10 {
11     cin>>n;
12     for(int i=0;i<n;i++)
13         cin>>p[i].x>>p[i].y;
14     for(int i=0;i<n;i++)
15     {
16         int tot=0;
17         for(int j=0;j<n;j++)
18         {
19             if(i==j)
20                 continue;
21             else if(p[i].x>=p[j].x&&p[i].y>=p[j].y)
22                 tot++;
23         }
24         ans[tot]++;
25     }
26     for(int i=0;i<n;i++)
27         cout<<ans[i]<<endl;
28     return 0;
29 }

其实呢,此题貌似用树状数组也行啊,只能说数据太水qwq!

树状数组,对X排下序之后以每个点的y进行操作。如果在某次算出sum(y)之后,肯定就是他的等级。因为在它之前的点都是x比它小或者和它相等,但是y没他大的。

 1 #include <bits/stdc++.h>
 2 using namespace std;
 3 const int N=16666;
 4 struct Node
 5 {
 6     int x,y;
 7 }p[N<<1];
 8 int n,c[N<<1],ans[N<<1];
 9 int lowbit(int x)
10 {
11     return x&-x;
12 }
13 inline bool cmp(Node p,Node q)
14 {
15     return p.x==q.x?(p.y<q.y):(p.x<q.x);
16 }
17 int sum(int x)
18 {
19     int ret=0;
20     while(x>0)
21     {
22         ret+=c[x];
23         x-=lowbit(x);
24     }
25     return ret;
26 }
27 void add(int x)
28 {
29     while(x<=32001)
30     {
31         c[x]++;
32         x+=lowbit(x);
33     }
34 }
35 int main()
36 {
37     cin>>n;
38     for(int i=1;i<=n;i++)
39         cin>>p[i].x>>p[i].y;
40     sort(p+1,p+1+n,cmp);
41     for(int i=1;i<=n;i++)
42     {
43         ans[sum(p[i].y)]++;
44         add(p[i].y);
45     }
46     for(int i=0;i<n;i++)
47         cout<<ans[i]<<endl;
48     return 0;
49 }
时间: 2024-09-30 18:10:02

Vijos P1066 弱弱的战壕【多解,线段树,暴力,树状数组】的相关文章

[FZYZOJ 1003] 弱弱的战壕

P1003 -- 弱弱的战壕 时间限制:1000MS 内存限制:65536KB Description 永恒和lmx正在玩一个即时战略游戏,名字嘛~~~~~~恕本人记性不好,忘了-_-b. mx在他的基地附近建立了n个战壕,每个战壕都是一个独立的作战单位,射程可以达到无限(“lmx不赢定了?!?”永恒[email protected][email protected]). 但是,战壕有一个弱点,就是只能攻击它的左下方,说白了就是横纵坐标都不大于它的点(mx:“我的战壕为什么这么菜”ToT).这样

弱弱的战壕

弱弱的战壕(jdoj1347-vijos1199) 题目大意:给你n个战壕的坐标,每个战壕只能保护其左下方的战壕 ( 明白为什么是弱弱的战壕了吧 ) ,最后输出n行,第 i 行表示保护了i-1个战壕的战壕个数. 注释:n<=15000 每一个战壕的坐标 ( x ,y ) ,x , y<=32000 想法:和清点人数一样,我们也可以通过线段树来维护,对应的,我们选择树状数组 ( 连修改都没有,扯什么扯 ) .怎么维护呢?我们思考...思考.......其实,我开始的想法是读入,然后统一处理,结果

lua的弱弱引用表

lua有GC,细节无需太关注,知道些基本的就行,能local就一定不要global: 还有在数组里的对象,除非显式=nil,否则很难回收: 不过可以用弱引用表来告诉GC.外部引用为0,就不要管我,请del it. weak table是通过元表实现,元表里的__mode字段包含k或者v:k表示key为弱引用:v表示value为弱引用. 1.首先看一个普通的例子: a = {} key = {} a[key] = 1 key = {} a[key]=2 collectgarbage() for k

java学习笔记-1205班李夏*:弱弱的说一句:尚硅谷,你值得拥有

时间过得真快,一晃眼一个月都过去了.记得前几个月看尚硅谷java视频的情景,还很清晰. 在这一个月中从最开始来北京java培训时的激动到现在的对北京的适应感觉经历了许多许多,再这一个月中我对北京的人文文化有了比较深入的了解,发现原来一个团队,集体可以如此和谐,每个人也都可以如此努力的为梦想,为工作,使我有了莫名的动力. 对于前一个阶段的总结,我主要可以分为以下几个方面: 第①方面:在学习上的总结,由于学校的期末考试我不得已把自己来北京的时间推迟了两周,来了之后发现这里的进度特别快,心想这下死定了

丑数[弱弱版]

丑数[弱弱版] Description 对于一给定的素数集合 S = {p1, p2, ..., pK}, 来考虑那些质因数全部属于S 的数的集合. 这个集合包括:p1, p1p2, p1p1, 和 p1p2p3 (还有其它).这是个对于一个输入的S的丑数集 合.注意:我们不认为1 是一个丑数. 你的工作是对于输入的集合S去寻找集合中的第N个丑数.longint(signed 32-bit)对于 程序是足够的. Input 第 1 行: 二个被空隔分开的整数:K 和 N , 1<= K<=10

Vijos P1448 校门外的树【多解,线段树,树状数组,括号序列法+暴力优化】

校门外的树 描述 校门外有很多树,有苹果树,香蕉树,有会扔石头的,有可以吃掉补充体力的…… 如今学校决定在某个时刻在某一段种上一种树,保证任一时刻不会出现两段相同种类的树,现有两个操作: K=1,K=1,读入l.r表示在区间[l,r]中种上一种树,每次操作种的树的种类都不同 K=2,读入l,r表示询问l~r之间能见到多少种树 (l,r>0) 格式 输入格式 第一行n,m表示道路总长为n,共有m个操作 接下来m行为m个操作 输出格式 对于每个k=2输出一个答案 样例1 样例输入1 5 4 1 1

一维 + 二维树状数组 + 单点更新 + 区间更新 详解

树状数组详解: 假设一维数组为A[i](i=1,2,...n),则与它对应的树状数组C[i](i=1,2,...n)是这样定义的: C1 = A1 C2 = A1 + A2 C3 = A3 C4 = A1 + A2 + A3 + A4 C5 = A5 C6 = A5 + A6 ................. C8 = A1 + A2 + A3 + A4 + A5 + A6 + A7 + A8 ................ 如图可知: 为奇数的时候他是代表他本身,而为偶数的时候则是代表着自

BIT 树状数组 详解 及 例题

(一)树状数组的概念 如果给定一个数组,要你求里面所有数的和,一般都会想到累加.但是当那个数组很大的时候,累加就显得太耗时了,时间复杂度为O(n),并且采用累加的方法还有一个局限,那就是,当修改掉数组中的元素后,仍然要你求数组中某段元素的和,就显得麻烦了.所以我们就要用到树状数组,他的时间复杂度为O(lgn),相比之下就快得多.下面就讲一下什么是树状数组: 一般讲到树状数组都会少不了下面这个图: 下面来分析一下上面那个图看能得出什么规律: 据图可知:c1=a1,c2=a1+a2,c3=a3,c4

高级数据结构:优先队列、图、前缀树、分段树以及树状数组详解

优秀的算法往往取决于你采用哪种数据结构,除了常规数据结构,日常更多也会遇到高级的数据结构,实现要比那些常用的数据结构要复杂得多,这些高级的数据结构能够让你在处理一些复杂问题的过程中多拥有一把利器.同时,掌握好它们的性质以及所适用的场合,在分析问题的时候回归本质,很多题目都能迎刃而解了. 这篇文章将重点介绍几种高级的数据结构,它们是:优先队列.图.前缀树.分段树以及树状数组. 一.优先队列 1.优先队列的作用 优先队列最大的作用是能保证每次取出的元素都是队列中优先级别最高的,这个优先级别可以是自定