zoukankan      html  css  js  c++  java
  • LOJ6435 PKUSC2018 星际穿越

    这个题吧当时在考场只得了45分 然后70分的性质都分析到了 不知道为啥就是写萎蛋了

    哎 当时还是too young too simple

    看了一下julao们的博客这个题有两种做法

    一个是比较费脑子的倍增做法

    一个是比较费体力【大雾 的主席树做法

    打死也不写数据结构的我当然还是学第一个啦

    首先我们可以分析到要么往右跳一步再往左跳 要么一直往左跳

    这个过程我们可以O(n^2)做 通过更新mn[x]就是每个点往左跳最远的位置 这样就可以一步一步往左跳

    然后这个过程可以优化一下就是从右往左扫一遍更新先往右跳的 比较好写233

    然后我们可以发现这个mn它是一段一段的 所以我们一步一步跳很浪费时间

    可以联想到倍增 也就是把一步一步换成2^i步这样 这样可以lgn换掉一个n

    所以我们可以用f[x][i]表示x走2^i步最远的位置

    我们再来观察题目性质

    它每次询问的是x走到一段区间的距离和 这个问题显然是可以差分的

    也就是x->[l,r] = x->[l,x] - x->[r+1,x]

    我们还可以观察到这个跳跃正反是一样的 于是我们要做的就是求calc(l,x) - calc(r+1,x)

    那么我们可以预处理一个s[x][i]数组表示x~f[x][i]这些点跳到f[x][i]的最少步数和

    所以我们calc要做的就是[l,x]这些点跳到x的步数和 也可以反过来做就是x象征性的往左跳 然后实际上统计的都是这些点到x的距离【有点拗口 但好像就是这么理解的

    然后跳的话。。。就是统计每次2^j跳的和然后加进来就行了

    有些小trick放在代码里了 不影响正常理解应该 只是大大缩短了代码量

    //Love and Freedom.
    #include<cstdio>
    #include<cmath>
    #include<algorithm>
    #include<cstring>
    #define ll long long
    #define inf 20021225
    #define N 300010
    #define LG 20
    using namespace std;
    
    int s[N][LG],f[N][LG];
    int w[N],n,l,r,x,Q;
    int gcd(int x,int y){return y?gcd(y,x%y):x;}
    ll calc(int l,int r)
    {
        if(w[r]<=l)    return r-l;
        ll ans=r-w[r]; r=w[r]; int tot=1;
        for(int i=LG-1;~i;i--)    if(f[r][i]>l)
            ans+=s[r][i]+tot*(r-f[r][i]),r=f[r][i],tot+=1<<i;
        return ans+(r-l)*(tot+1);
    }
    int main()
    {
        scanf("%d",&n); w[1]=1;
        for(int i=2;i<=n;i++)    scanf("%d",&w[i]);
        f[n][0]=w[n]; s[n][0]=n-w[n];
        for(int i=n-1;i;i--)    f[i][0]=min(f[i+1][0],w[i]),s[i][0]=i-f[i][0];
        for(int i=1;i<LG;i++)    for(int j=1;j<=n;j++)    if(f[j][i-1])
            f[j][i]=f[f[j][i-1]][i-1],
            s[j][i]=s[j][i-1]+s[f[j][i-1]][i-1]+(f[j][i-1]-f[j][i])*(1ll<<(i-1));
        scanf("%d",&Q);
        while(Q--)
        {
            scanf("%d%d%d",&l,&r,&x);
            ll tp=calc(l,x)-calc(r+1,x),dn=r-l+1;
            ll g=gcd(tp%dn,dn);
            printf("%lld/%lld
    ",tp/g,dn/g);
        }
        return 0;
    }
    View Code
  • 相关阅读:
    ASP.NET AJAX Sys未定义的解决方法
    幽默中的人生哲理
    我们总是试图抹去一些东西,比如记忆。然而,一切只是徒然。
    在webbrowser中操作js弹出的对话框 以及触发其它事件····
    Asp.Net :写入 、读取 、 删除、追加cookie数组?
    揭开ASP.NET中Cookie编程的奥秘
    多线程调度例子·····················
    winform编程——做过ini配置文件创建文件、添加配置项,修改配置项,删除配置项
    C# 比较二个时间差。Environment.TickCount
     WebBrowser控件捕捉DHTML事件
  • 原文地址:https://www.cnblogs.com/hanyuweining/p/11043306.html
Copyright © 2011-2022 走看看