我是靠谱客的博主 耍酷大地,最近开发中收集的这篇文章主要介绍【LOJ】#6435. 「PKUSC2018」星际穿越,觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

题解

想出70的大众分之后就弃疗了,正解有点神仙

就是首先有个比较显然的结论,就是要么是一直往左走,要么是走一步右边,然后一直往左走

根据这个可以结合RMQ写个70分的暴力

我们就考虑,最优的话显然是走一步左边就到了目标点,第二步才开始有分叉
假如我们先走了一步左边,然后就变成了,从(L[x])开始走,下一步可以走到([L[x],N])的所有点最小的转移点之前,之后再把后来走的点代价都加上1即可
这样的话,不管是一直走左边,还是走了一步右边再走了左边,情况都被包含了

这个时候考虑这个问题就比较简单了,可以使用倍增
(f[i][j])表示([i,n])内最小的(l[x])的值
(s[i][j])表示(i)走到(f[i][j])内所有点的距离和

转移就是
(f[i][j] = f[f[i][j - 1]][j - 1])
(s[i][j] = s[i][j - 1] + s[f[i][j - 1]][j - 1] + 2^{j - 1} * (f[i][j - 1] - f[i][j]))

查询两端前缀和,查的时候直接把(x)变成(L[x])进行倍增即可

代码

#include <bits/stdc++.h>
#define fi first
#define se second
#define pii pair<int,int>
#define pdi pair<db,int>
#define mp make_pair
#define pb push_back
#define enter putchar('n')
#define space putchar(' ')
#define eps 1e-8
#define mo 974711
#define MAXN 300005
//#define ivorysi
using namespace std;
typedef long long int64;
typedef double db;
template<class T>
void read(T &res) {
    res = 0;char c = getchar();T f = 1;
    while(c < '0' || c > '9') {
    if(c == '-') f = -1;
    c = getchar();
    }
    while(c >= '0' && c <= '9') {
    res = res * 10 + c - '0';
    c = getchar();
    }
    res *= f;
}
template<class T>
void out(T x) {
    if(x < 0) {x = -x;putchar('-');}
    if(x >= 10) {
    out(x / 10);
    }
    putchar('0' + x % 10);
}
int N,L[MAXN];
int f[MAXN][20];
int64 s[MAXN][20];
void Init() {
    read(N);
    for(int i = 2 ; i <= N ; ++i) read(L[i]);
    f[N][0] = L[N];s[N][0] = N - L[N];
    for(int i = N - 1 ; i >= 1 ; --i) {
    f[i][0] = min(f[i + 1][0],L[i]);s[i][0] = i - f[i][0];
    }
    for(int j = 1 ; j <= 19 ; ++j) {
    for(int i = 1 ; i <= N ; ++i) {
        f[i][j] = f[f[i][j - 1]][j - 1];
        s[i][j] = s[i][j - 1] + s[f[i][j - 1]][j - 1] + 1LL * (f[i][j - 1] - f[i][j]) * (1 << j - 1); 
    }
    }
}
int64 gcd(int64 a,int64 b) {
    return b == 0 ? a : gcd(b,a % b);
}
int64 Calc(int tar,int st) {
    if(tar >= L[st]) return st - tar;
    int64 res = st - L[st];st = L[st];
    int64 sum = 1;
    for(int j = 19 ; j >= 0 ; --j) {
    if(f[st][j] >= tar) {
        res += s[st][j];
        res += 1LL * sum * (st - f[st][j]);
        st = f[st][j];
        sum += 1 << j;
    }
    }
    res += 1LL * (sum + 1) * (st - tar);
    return res;
}
void Solve() {
    int Q;int l,r,x;
    read(Q);
    while(Q--) {
    read(l);read(r);read(x);
    int64 u = Calc(l,x) - Calc(r + 1,x),d = r - l + 1,g = gcd(u,d);
    u /= g;d /= g;
    out(u);putchar('/');out(d);enter;
    }
}
int main() {
#ifdef ivorysi
    freopen("f1.in","r",stdin);
#endif
    Init();
    Solve();
}

转载于:https://www.cnblogs.com/ivorysi/p/10122578.html

最后

以上就是耍酷大地为你收集整理的【LOJ】#6435. 「PKUSC2018」星际穿越的全部内容,希望文章能够帮你解决【LOJ】#6435. 「PKUSC2018」星际穿越所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(44)

评论列表共有 0 条评论

立即
投稿
返回
顶部