考虑到我们已经排号按照什么顺序去浏览
考虑到贡献位
∑
m
a
x
(
c
i
,
a
j
−
a
i
)
\sum max(c_i,a_j-a_i)
∑max(ci,aj−ai),转换成
∑
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,aj−ai−ci)
转化位规定第
i
i
i号定义为
[
a
i
,
a
i
+
c
i
]
[a_i,a_i+c_i]
[ai,ai+ci]这个区间
i
→
j
i \to j
i→j就是从
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;
}