Sicly今晚又有周赛了,兴致勃勃地找了一道题来做,却调了几个钟都找不到一个错误的数据,但总是返回Wrong Answer!只好贴到此处,请教各位大牛。
题目大意
给出平面上n(n<=105)个点的X、Y坐标(xi,yi)(0≤xi,yi≤109)及权值pi,求一条折线,满足:
1)折线只能向X正方向或Y正方向延伸;
2)折线的权值定义为折线上所有点的权值之和。
输出权值最大的折线的权值。
我的做法
先离散化,然后将n个点按第一顺序为X升序,第二顺序为Y升序来排序。
再建一棵线段树,每读一个项Pi =(xi, yi, pi);就在线段树里查询pi点的颜色C(yi),并加一条线段(yi,+∞),使得线段上每一点q的颜色C(qi)=max(C(yi)+pi,C(qi))。 最后输出C(+∞)即可。时间复杂度O(NlogN),空间复杂度O(N)。代码如下:
#include
<
cstdio
>
#include
<
cstring
>
#include
<
algorithm
>
#include
<
iostream
>
using
namespace
std;
const
int
N
=
100010
;
//
车站数
int
rx[N],ry[N],p[N],x[N],y[N],xp,yp;
struct
S
{
//
一个车站
int
x,y,p;
bool
operator
<
(
const
S
&
o)
const
{
return
x
==
o.x
?
y
<
o.y : x
<
o.x;
}
}
t[N];
template
<
int
N
>
class
PntTree
{
//
线段树的一种实现
int
a[
2
*
N];
public
:
/**/
/*
断言: i∈[0,N-1) (N-1用于表示+∞)
*/
void
add(
int
i,
int
c)
{
//
添加点i
c
+=
C(i);
for
(a[i
+=
N]
=
c; i
>
1
; i
/=
2
)
if
(
~
i
&
1
)
if
(c
>
a[i
/
2
]) a[i
/
2
]
=
c;
else
return
;
}
int
C(
int
i)
{
//
返回点i的颜色
int
c
=
a[i
+
N];
for
(i
+=
N; i
>
1
; i
/=
2
)
if
(i
&
1
) c
=
max(c,a[i
/
2
]);
return
c;
}
}
;
PntTree
<
131072
>
tr;
int
main()
{
int
i,j,n,tx,ty;
scanf(
"
%d%d%d
"
,
&
tx,
&
ty,
&
n);
//
离散化
xp
=
yp
=
0
;
for
(i
=
0
; i
<
n;
++
i)
{
scanf(
"
%d%d%d
"
,
&
rx[i],
&
ry[i],
&
p[i]);
x[xp
++
]
=
rx[i];
y[yp
++
]
=
ry[i];
}
sort(x,x
+
xp);
sort(y,y
+
yp);
xp
=
unique(x,x
+
xp)
-
x;
yp
=
unique(y,y
+
yp)
-
y;
for
(i
=
0
; i
<
n;
++
i)
{
t[i].x
=
lower_bound(x,x
+
xp,rx[i])
-
x;
t[i].y
=
lower_bound(y,y
+
yp,ry[i])
-
y;
t[i].p
=
p[i];
}
//
排序
sort(t,t
+
n);
//
线段树操作
for
(i
=
0
; i
<
n;
++
i)
{
//
cout<<t[i].x<<' '<<t[i].y<<' '<<t[i].p<<' '<<tr.C(t[i].y)<<"->";
tr.add(t[i].y, t[i].p);
//
cout<<tr.C(t[i].y)<<endl;
}
//
输出无穷远处的颜色
printf(
"
%d\n
"
,tr.C(N
-
1
));
return
0
;
}
附:题目全文
Task: The Bus
The streets of Byte City form a regular, chessboardlike network - they are either north-south or west-east directed. We shall call them NS- and WE-streets. Furthermore, each street crosses the whole city. Every NS-street intersects every WE- one and vice versa. The NS-streets are numbered from 1 to n, starting from the westernmost. The WE-streets are numbered from 1 to m, beginning with the southernmost. Each intersection of the i-th NS-street with the j-th WE-street is denoted by a pair of numbers (i, j) (for 1 <= i <= n, 1 <= j <= m).
There is a bus line in Byte City, with intersections serving as bus stops. The bus begins its itinerary by the (1, 1) intersection, and finishes by the (n, m) intersection. Moreover, the bus may only travel in the eastern and/or northern direction.
There are passengers awaiting the bus by some of the intersections. The bus driver wants to choose his route in a way that allows him to take as many of them as possible. (We shall make an assumption that the interior of the bus is spacious enough to take all of the awaiting passengers, regardless of the route chosen.)
Task
Write a programme which:
- reads from the standard input a description of the road network and the number of passengers waiting at each intersection,
- finds, how many passangers the bus can take at the most,
- writes the outcome to the standard output.
Input
The first line of the standard input contains three positive integers n, m and k -- denoting the number of NS-streets, the number of WE-streets and the number of intersections by which the passengers await the bus, respectively. (1 <= n <= 109, 1 <= m <= 109, 1 <= k <= 105).
The following k lines describe the deployment of passengers awaiting the bus, a single line per intersection. In the i + 1st line there are three positive integers xi, yi and pi, separated by single spaces, 1 <= xi <= n, 1 <= yi <= m, 1 <= pi <= 106. A triplet of this form signifies that by the intersection (xi, yi)pi passengers await the bus. Each intersection is described in the input data once at the most. The total number of passengers waiting for the bus does not exceed 1 000 000 000.
Output
Your programme should write to the standard output one line containing a single integer - the greatest number of passengers the bus can take.
Example
For the input data:
8 7 11
4 3 4
6 2 4
2 3 2
5 6 1
2 5 2
1 5 5
2 1 1
3 1 1
7 7 1
7 4 2
8 6 2
the correct outcome is:
11
题目来源
posted on 2006-12-09 22:23
踏雪赤兔 阅读(733)
评论(3) 编辑 收藏 引用 所属分类:
玩转编程