28届宁波市中小学生程序设计竞赛复赛(初中组)

第4题-迷宫大门

来源于宁波市程序设计竞赛复赛(初中组)
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;
}

你可能感兴趣的:(杂题)