题意
给出n条线段的左右端点的坐标 [ l , r ] [l,r] [l,r],以及对应的价格c。 购买这n条线段后,会获得最少覆盖一次的点对应的数字,以及位于这些线段中间但未覆盖的这些数字。有一些原因 ,商店只出售前 s ( 1 ≤ s ≤ n ) s(1 \leq s \leq n) s(1≤s≤n)条线段。你需要从前s段中选出若干条线段,使得获得的数字尽可能地多,同时尽量使总价格最小。
思路
记录三条线段:
x
1
:
左
端
点
最
小
且
价
值
最
小
的
线
段
x_1: 左端点最小且价值最小的线段
x1:左端点最小且价值最小的线段
x
2
:
右
端
点
最
大
且
价
值
最
小
的
线
段
x_2: 右端点最大且价值最小的线段
x2:右端点最大且价值最小的线段
x
3
:
长
度
最
长
且
价
值
最
小
的
线
段
x_3: 长度最长且价值最小的线段
x3:长度最长且价值最小的线段
记录所得数字的最大区间
[
l
,
r
]
[l, r]
[l,r]
注意
写题时没有想到设置x3。若没有x3会出现下面的bug(购买c,而不购买a + b)。
需要考虑这样的特殊情况:
|---------------------------------------------| 价值c
|----------|
\qquad\qquad\qquad\quad
|--------| 价值a,b
代码
#include<bits/stdc++.h>
using namespace std;
const int N = 2e5 + 7, M = 1e6;
struct node
{
int l, r, c;
}s[N];
int main()
{
int T; cin >> T;
while (T--)
{
int n; cin >> n;
for (int i = 1; i <= n; i++)
cin >> s[i].l >> s[i].r >> s[i].c;
//x1:左边界(l)最小且最便宜的,x2右边界(r)最大且最便宜的
//x3:长度(maxn)最长且最便宜的
int l = s[1].l, r = s[1].r, maxn = s[1].r - s[1].l + 1;
int x1 = 1, x2 = 1, x3 = 1;
for (int i = 1; i <= n; i++)
{
//x3
if (s[i].r - s[i].l + 1 > maxn) maxn = s[i].r - s[i].l + 1, x3 = i;
else if (s[i].r - s[i].l + 1 == maxn && s[i].c < s[x3].c) x3 = i;
//x1
if (s[i].l < l) x1 = i, l = s[i].l;
else if (s[i].l == l && s[i].c < s[x1].c) x1 = i;
//x2
if (s[i].r > r) x2 = i, r = s[i].r;
else if (s[i].r == r && s[i].c < s[x2].c) x2 = i;
//cout << x1 << "---" << x2 << "---" << x3 << endl;
int sum;
if (x1 == x2) sum = s[x1].c;
else sum = s[x1].c + s[x2].c;
//特判
if (s[x3].l == s[x1].l && s[x3].r == s[x2].r && s[x3].c < sum)
sum = s[x3].c;
cout << sum << endl;
}
}
return 0;
}