来源于宁波市程序设计竞赛复赛(初中组)
n≤500,000,显然对于这么大的n,O(n^2)的解法都不够。那么考虑O(n)或者O(n*lg(n))的解法,动态规划?貌似可行,但状态怎么表示?左对齐?右对齐?除此,要怎么进行转移呢,毕竟a和b的范围也相当大,即便进行hash,最多也有500,000?
继续分析发现,题目中指出,每一处对齐都只得1分,这说明在左右不可能同时对齐的情况下,左对齐和右对齐的收益是一样的。那我们不妨让所有小球都尽量左对齐好了,直到某一处不能左对齐便从后一个开始继续左对齐,反证法可以证明这样能得到最大收益。
对齐实际上是保持高度相等,例如对于第一根绳子上的右小球,可以保持的高度范围是[0,a1+b1],第二根绳子的左小球可以保持的高度范围是[0,a2+b2]。
要是这两个小球保持高度相等,那么我们必须保证他们的高度在[0,min(a1+b1,a2+b2)]范围内,相应地,第二根绳子的右小球变化范围就必须是[a2+b2-min(a1+b1,a2+b2),a2+b2]…..依次类推,每两个相邻小球可以对齐便计1分,不能对齐便跳过,下一个继续对齐。
#include
#include
#include
#include
#include
using namespace std;
const int maxn=5e5+15;
int l[maxn], r[maxn];
bool ok(int s1,int t1,int s2,int t2){
int s=max(s1,s2);
int t=min(t1,t2);
if(s<=t) return true;
return false;
}
int main()
{
int n;
freopen("door.in","r",stdin);//从in.txt中读取数据
freopen("door.out","w",stdout);//输出到out.txt文件
scanf("%d",&n);
for(int i=1;i<=n;i++)
scanf("%d%d",&l[i],&r[i]);
int ma=l[1]+r[1], mi=0;
int ans=0;
for(int i=2;i<=n;i++){
if(ok(mi,ma,0,l[i]+r[i])){
ans++;
int len=l[i]+r[i];
int tmi=max(0,len-ma);
int tma=max(0,len-mi);
mi=tmi;
ma=tma;
}else{
mi=0;
ma=l[i]+r[i];
}
// printf("%d %d %d\n",ans, mi, ma);
}
printf("%d\n",ans);
return 0;
}