2018中国大学生程序设计竞赛 - 网络选拔赛 1010 YJJ's Salesman 【离散化+树状数组维护区间最大值】

时间:2021-01-07 12:46:34

题目传送门:http://acm.hdu.edu.cn/showproblem.php?pid=6447

YJJ's Salesman

Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)
Total Submission(s): 919    Accepted Submission(s): 290

Problem Description

YJJ is a salesman who has traveled through western country. YJJ is always on journey. Either is he at the destination, or on the way to destination.
One day, he is going to travel from city A to southeastern city B. Let us assume that A is (0,0) on the rectangle map and B (109,109). YJJ is so busy so he never turn back or go twice the same way, he will only move to east, south or southeast, which means, if YJJ is at (x,y) now (0≤x≤109,0≤y≤109), he will only forward to (x+1,y), (x,y+1) or (x+1,y+1).
On the rectangle map from (0,0) to (109,109), there are several villages scattering on the map. Villagers will do business deals with salesmen from northwestern, but not northern or western. In mathematical language, this means when there is a village k on (xk,yk) (1≤xk≤109,1≤yk≤109), only the one who was from (xk−1,yk−1) to (xk,yk) will be able to earn vk dollars.(YJJ may get different number of dollars from different village.)
YJJ has no time to plan the path, can you help him to find maximum of dollars YJJ can get.

Input

The first line of the input contains an integer T (1≤T≤10),which is the number of test cases.
In each case, the first line of the input contains an integer N (1≤N≤105).The following N lines, the k-th line contains 3 integers, xk,yk,vk (0≤vk≤103), which indicate that there is a village on (xk,yk) and he can get vk dollars in that village.
The positions of each village is distinct.
 

Output

The maximum of dollars YJJ can get.
 
Sample Input
1
3
1 1 1
1 2 2
3 3 1
Sample Output
3
 
Source

题目大意:

有N个村庄 [ 横纵坐标为 x , y ( x , y >= 1 && x , y <=1e9 ) ] , 每到一个村庄 我们可以选择向上向右或者向右上方走,每个村庄有一个贡献值,只有当我是从左下方的村庄来的才可以得到当前村庄的贡献值。

解题思路:

离散化 +排序+树状数组维护区间最大值

坐标 x, y 的范围太大,而节点范围还可以,所以不妨离散化一波(听说实际测试数据范围没有到1e9,出题人就料定我们没有这个胆)。

一开始用传统 dp ,状态转移很好想,但很明显会超时,交了两波TLE果断自闭。

这道题是正确打开方式是线段树或者树状数组维护区间最大值,而这里的区间是 y 的区间。然后通过前一行的最优值去转移得到当前行的最优值,不过排序顺序需要注意,因为只有左下方的可以转移过来所以当 x 相同时,优先处理 y 大的。

AC code:

 #include <bits/stdc++.h>
#define ll long long int
#define INF 0x3f3f3f3f
#define mod 1000000007
using namespace std; const int MAXN = 1e5+; struct date{
int x, y, v;
}node[MAXN]; int d[MAXN], dp[MAXN];
int t[MAXN<<];
int N; bool cmp1(struct date a, struct date b)
{
if(a.x == b.x) return a.y > b.y;
return a.x < b.x;
}
int lowbit(int x)
{
return x&(-x);
}
void add(int st, int val)
{
for(int i = st; i <= MAXN; i+=lowbit(i))
t[i] = max(t[i], val);
}
int query(int st)
{
int res = ;
for(int i = st; i > ; i-=lowbit(i))
res = max(res, t[i]);
return res;
} int main()
{
int T;
scanf("%d", &T);
while(T--){
scanf("%d", &N);
memset(t, , sizeof(t));
for(int i = ; i <= N; i++){
scanf("%d%d%d", &node[i].x, &node[i].y, &node[i].v);
d[i] = node[i].y;
}
sort(d+, d+N+);
int last = unique(d+, d++N)--d;
for(int i = ; i <= N; i++)
node[i].y = lower_bound(d+, d++last, node[i].y)-d;
sort(node+, node++N, cmp1);
for(int i = ; i <= N; i++)
{
int res = query(node[i].y-) + node[i].v;
add(node[i].y, res);
}
printf("%d\n", query(MAXN-));
}
return ;
}