B - Integers Shop(贪心 + 思维)

Hello 2022(codeforces)

题意

给出n条线段的左右端点的坐标 [ l , r ] [l,r] [lr],以及对应的价格c。 购买这n条线段后,会获得最少覆盖一次的点对应的数字,以及位于这些线段中间但未覆盖的这些数字。有一些原因 ,商店只出售前 s ( 1 ≤ s ≤ n ) s(1 \leq s \leq n) s(1sn)条线段。你需要从前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;
}
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

to cling

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值