[CF1503C]Travelling Salesman Problem

考虑到我们已经排号按照什么顺序去浏览
考虑到贡献位 ∑ m a x ( c i , a j − a i ) \sum max(c_i,a_j-a_i) max(ci,ajai),转换成 ∑ c i + ∑ m a x ( 0 , a j − a i − c i ) \sum c_i+\sum max(0,a_j-a_i-c_i) ci+max(0,ajaici)
转化位规定第 i i i号定义为 [ a i , a i + c i ] [a_i,a_i+c_i] [ai,ai+ci]这个区间
i → j i \to j ij就是从 i i i区间到 j j j区间的横向最短挪动,显然按左端点排序后最优,从最大的右端点前往,如果左端点不是最大的,那一定包括了最大的左端点的那个区间,不然他会往回跳就直接GG了
于是有了简单的代码

#include <bits/stdc++.h>
#define pii pair<int,int>
using namespace std;
typedef long long LL;
const int maxn = 2e5+5;
int readint(){
	int x=0,f=1;char s=getchar();
	#define sc (s=getchar())
	while(s<'0'||s>'9'){
		if(s=='-')
			f=-1;
		sc;
	}
	while(s>='0'&&s<='9'){
		x=(x<<3)+(x<<1)+(s^48);
		sc;
	}
	#undef sc
	return x*f;
}
struct zz{
	int a,c;
}s[maxn]; 
bool cmp(const zz &x,const zz &y){
	return x.a<y.a;
}
int main (){
	int n=readint();
	for(int i=1;i<=n;i++)
		s[i].a=readint(),s[i].c=readint();
	sort(s+1,s+n+1,cmp);
	int maxx=s[1].a+s[1].c;
	LL ans=s[1].c;
	for(int i=2;i<=n;i++){
		ans=ans+max(0,s[i].a-maxx);
		maxx=max(maxx,s[i].a+s[i].c);
		ans+=s[i].c;
	}
	cout<<ans<<endl;
	return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值