2019 ICPC North American Qualifier Contest L题 Traveling Merchant

题意

一条线上有 N ( ≤ 1 0 5 ) N(\leq10^5) N(105)个城市,标号从 1 1 1 N N N。每个城市物价根据星期的不同而有所不同,具体来说,每个城市有原价 v i v_i vi,变动 d i d_i di,周一到周日的变化量为 { + 0 , + v i , + 2 v i , + 3 v i , + 2 v i , + v i , + 0 } \{+0, +v_i, +2v_i, +3v_i, +2v_i, +v_i, +0\} {+0,+vi,+2vi,+3vi,+2vi,+vi,+0}。现在有 Q ( ≤ 1 0 5 ) Q(\leq 10^5) Q(105)个询问,每个询问商人会从 l l l号城市往 r r r号城市旅行, l l l有可能大于 r r r,此时要逆行。求次旅行最大的差价 ( p r i c e j − p r i c e i ) ( j > i ) (price_j-price_i) (j > i) (pricejpricei)(j>i)是多少。

解题思路

1 0 5 10^5 105数据量+ 1 0 5 10^5 105查询,那么很明显直接模拟是不行的,但是这题题意看起来用线段树也不太好维护(有可能是我太菜了不会)。于是有没有又好维护跑的又快的数据结构呢?于是祭出分块大法。
我们可以把所有城市按照标号分成 N \sqrt{N} N 块,每一块内存储3个信息 { 整 块 内 最 大 价 格 , 整 块 内 最 小 价 格 , 整 块 内 最 大 差 价 } \{整块内最大价格,整块内最小价格,整块内最大差价\} {}。这样查询的时候可以两端直接模拟,中间根据这三个信息更新最大差价,每次查询复杂度不会超过 O ( N ) O(\sqrt{N}) O(N )
由于有星期这个设定,所以我们要维护 7 7 7重块,每个块信息代表当从第 k k k天开始从左到右(反向旅行同理)旅行的信息。这个问题便迎刃而解了,看官方题解好像是用的线段树,但是感觉应该很难写,我用分块一发就过了,线段树不知道要调多久。

时间复杂度

O ( N + Q N ) O(N+Q\sqrt{N}) O(N+QN )

代码

#include <algorithm>
#include <cmath>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <map>
#include <queue>
#include <set>
#include <stack>
#include <vector>

typedef long long ll;
typedef unsigned long long ull;
using namespace std;
const int INF = 2147483647;
const int INF2 = 0x3f3f3f3f;
const ll INF64 = 1e18;
const double INFD = 1e30;
const double EPS = 1e-6;
const double PI = 3.14159265;
const ll MOD = 1e9 + 7;

template <typename T>
inline T read() {
    T X = 0, w = 0;
    char ch = 0;
    while (!isdigit(ch)) {
        w |= ch == '-';
        ch = getchar();
    }
    while (isdigit(ch)) X = (X << 3) + (X << 1) + (ch ^ 48), ch = getchar();
    return w ? -X : X;
}

const int MAXN = 100005;
int n, m, k;

struct Info {
    int v, d, b;
};

// 块信息,最大值,最小值,最大差值,左右延伸范围
struct Block {
    int maxx;
    int minn;
    int maxprofit;
    int lft;
    int rgt;
};
int raise[7] = {0, 1, 2, 3, 2, 1, 0};
int CASE;
// 正向旅行块
Block blocks[7][505];
// 反向旅行块
Block blocks2[7][505];
int B;
Info cities[MAXN];

// 正向计算
void cal(int l, int r, int d, int& maxx, int& minn, int& maxprofit) {
    for (int j = l; j <= r; j++) {
        int day = (j - l + d) % 7;
        int price = cities[j].v + raise[day] * cities[j].d;
        minn = min(minn, price);
        maxprofit = max(maxprofit, price - minn);
        maxx = max(maxx, price);
    }
}

// 反向计算
void cal2(int r, int l, int d, int& maxx, int& minn, int& maxprofit) {
    for (int j = r; j >= l; j--) {
        int day = (r - j + d) % 7;
        int price = cities[j].v + raise[day] * cities[j].d;
        minn = min(minn, price);
        maxprofit = max(maxprofit, price - minn);
        maxx = max(maxx, price);
    }
}

// 同样也是一正一反
int query2(int r, int l) {
    int d = 0;
    int maxx = 0, minn = 1e9, prof = 0;
    int rs = max(l, blocks2[0][cities[r].b].lft);
    cal2(r, rs, d, maxx, minn, prof);
    d = (d + r - rs + 1) % 7;
    for (int i = cities[r].b - 1; i >= cities[l].b + 1; i--) {
        prof = max(prof, blocks2[d][i].maxx - minn);
        minn = min(minn, blocks2[d][i].minn);
        prof = max(prof, blocks2[d][i].maxprofit);
        maxx = max(maxx, blocks2[d][i].maxx);
        d = (d + blocks2[d][i].rgt - blocks2[d][i].lft + 1) % 7;
    }
    if (cities[l].b != cities[r].b) {
        int ls = min(r, blocks2[0][cities[l].b].rgt);
        cal2(ls, l, d, maxx, minn, prof);
    }
    return prof;
}

int query(int l, int r) {
    int d = 0;
    int ls = min(r, blocks[0][cities[l].b].rgt);
    int maxx = 0, minn = 1e9, prof = 0;

    cal(l, ls, d, maxx, minn, prof);
    d = (d + ls - l + 1) % 7;
    for (int i = cities[l].b + 1; i <= cities[r].b - 1; i++) {
        prof = max(prof, blocks[d][i].maxx - minn);
        minn = min(minn, blocks[d][i].minn);
        prof = max(prof, blocks[d][i].maxprofit);
        maxx = max(maxx, blocks[d][i].maxx);
        d = (d + blocks[d][i].rgt - blocks[d][i].lft + 1) % 7;
    }
    if (cities[l].b != cities[r].b) {
        int rs = max(l, blocks[0][cities[r].b].lft);
        cal(rs, r, d, maxx, minn, prof);
    }
    return prof;
}

int main() {
#ifdef LOCALLL
    freopen("in", "r", stdin);
    freopen("out", "w", stdout);
#endif
    scanf("%d", &n);
    B = (int)sqrt(n) + 1;
    int cnt = (n + B - 1) / B;
    for (int i = 1; i <= n; i++) {
        scanf("%d%d", &cities[i].v, &cities[i].d);
        cities[i].b = (i - 1) / B + 1;
    }
    for (int d = 0; d < 7; d++) {
        for (int i = 1; i <= cnt; i++) {
            // 确定块的左右边界
            blocks[d][i].lft = B * (i - 1) + 1;
            blocks[d][i].rgt = B * i;
            blocks2[d][i].lft = B * (i - 1) + 1;
            blocks2[d][i].rgt = B * i;
            int l = blocks[d][i].lft, r = blocks[d][i].rgt;
            Block& cur = blocks[d][i];
            cur.maxx = 0, cur.minn = 1e9, cur.maxprofit = 0;
            Block& cur2 = blocks2[d][i];
            cur2.maxx = 0, cur2.minn = 1e9, cur2.maxprofit = 0;
            cal(l, r, d, cur.maxx, cur.minn, cur.maxprofit);
            cal2(r, l, d, cur2.maxx, cur2.minn, cur2.maxprofit);
        }
    }
    scanf("%d", &m);
    while (m--) {
        int l, r;
        scanf("%d%d", &l, &r);
        if (l <= r) {
            printf("%d\n", query(l, r));
        } else {
            printf("%d\n", query2(l, r));
        }
    }
    return 0;
}
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值