树状数组维护DP——前缀最大值

树状数组维护DP

例题1

例题1

给定一个 n × m n\times m n×m 的网格图,设 ( i , j ) (i,j) (i,j) 表示从上往下数第 i i i 行与从左往右数第 j j j 列的交点。有一辆公交车在 ( 1 , 1 ) (1,1) (1,1) 处,它每次只能向往右或下行驶,目的地是 ( n , m ) (n, m) (n,m),给定 k k k 个公交站点,每个站点 ( x i , y i ) (x_i,y_i) (xi,yi) 有 p i p_i pi 位乘客,请问公交车至多能载多少名乘客到达终点?

数据范围: 1 ≤ n , m ≤ 1 0 9 , 1 ≤ k ≤ 1 0 5 1\le n,m\le 10^9,1\le k\le 10^5 1≤n,m≤109,1≤k≤105, 1 ≤ x i ≤ n 1\le x_i\le n 1≤xi≤n, 1 ≤ y i ≤ m 1\le y_i\le m 1≤yi≤m, 1 ≤ p i ≤ 1 0 6 1\le p_i\le 10^6 1≤pi≤106。

题解

如果 n , m n,m n,m 较小,我们可以利用动态规划求解: d p [ i ] [ j ] = max ⁡ ( d p [ i − 1 ] [ j ] , d p [ i ] [ j − 1 ] ) dp[i][j]=\max(dp[i-1][j],dp[i][j-1]) dp[i][j]=max(dp[i−1][j],dp[i][j−1]),时间复杂度是 O ( n m ) O(nm) O(nm)。

但 n , m n,m n,m 的规模较大,我们并不能遍历所有的格子进行转移,对于一个车站 ( i , j ) (i,j) (i,j),如果我们要求出从 ( 1 , 1 ) (1,1) (1,1) 到达 ( i , j ) (i,j) (i,j) 的最大乘客数量,应该从 ( i , j ) (i,j) (i,j) 的左上角里的最大值进行转移。

对所有的公交站点坐标 ( x , y ) (x,y) (x,y) 进行离散化,且将所有的公交站点按 y y y 从小到大排序, y y y 相同则按 x x x 从小到大排序。

设 d p [ x ] dp[x] dp[x] 表示大巴车从起点行驶到离散后的第 x x x 行能装载的最大乘客数量。

对于当前的 ( x i , y i ) (x_i,y_i) (xi,yi),所有在其左上角的 ( x j , y j ) (x_j,y_j) (xj,yj) 均已出现,所以 d p [ x i ] dp[x_i] dp[xi] 只需从 d p [ 1 ∼ x i ] dp[1\sim x_i] dp[1∼xi] 处挑选最大值进行转移。

而树状数组能够维护支持单点修改的前缀最大值,总时间复杂度为 O ( k log ⁡ k ) O(k\log k) O(klogk)。

值得注意的是,离散后的 x x x 值可能是 0 0 0,但是树状数组只能维护从 1 1 1 开始的前缀,所以需要将离散后的值偏移到从 1 1 1 开始。

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;
//#pragma GCC optimize(2)
#define int long long
#define endl '\n'
#define PII pair<int,int>
#define INF 1e18

struct Fenwick {
    int n;
    vector <int> tree;

    Fenwick(int _n) {
        n = _n;
        tree.assign(n + 2, 0);
    }

    int getPrefixMax(int x) {
        int maxn = 0;

        while (x) {
            maxn = max(maxn, tree[x]);
            x = x - lowbit(x);
        }
        return maxn;
    }

    void add (int x, int val) {
        while (x <= n) {
            tree[x] = max(tree[x], val);
            x += lowbit(x);
        }
    }

    static inline int lowbit(int x) {
        return x & (-x);
    }
};

struct node {
    int x, y;
    int p;
};

void lsh (vector <int> & a) {
    sort(a.begin(), a.end());
    a.erase(unique(a.begin(), a.end()), a.end());
}

void slove () {
    int n, m, k;
    cin >> n >> m >> k;

    vector <node> v;

    vector <int> xPos, yPos;
    for (int i = 1; i <= k; i++) {
        int x, y, p;
        cin >> x >> y >> p;
        xPos.push_back(x);
        yPos.push_back(y);
        v.push_back(node{x, y, p});
    }

    sort (v.begin(), v.end(), [](node A, node B) {
        if (A.y != B.y) return A.y < B.y;
        return A.x < B.x;
    });

    lsh(xPos);
    lsh(yPos);

    Fenwick tt(xPos.size());
    vector <int> dp(xPos.size(), 0);

    for (int i = 0; i < v.size(); i++) {
        int x;
        x = lower_bound(xPos.begin(), xPos.end(), v[i].x) - xPos.begin();
        int maxn;
        maxn = tt.getPrefixMax(x + 1);
        dp[x] = maxn + v[i].p;
        tt.add(x + 1, dp[x]);
    }

    int ans = 0;
    for (int i = 0; i < xPos.size(); i++) {
        ans = max(ans, dp[i]);
    }
    cout << ans << endl;
}

signed main () {
    ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
    slove();
}
相关推荐
Q741_1474 小时前
C++ 分治 归并排序 归并排序VS快速排序 力扣 912. 排序数组 题解 每日一题
c++·算法·leetcode·归并排序·分治
三体世界4 小时前
Qt从入门到放弃学习之路(1)
开发语言·c++·git·qt·学习·前端框架·编辑器
victory04314 小时前
K8S 安装 部署 文档
算法·贪心算法·kubernetes
月疯4 小时前
样本熵和泊松指数的计算流程!!!
算法·机器学习·概率论
机器学习之心4 小时前
MATLAB基于自适应动态特征加权的K-means算法
算法·matlab·kmeans
minji...5 小时前
算法题 逆波兰表达式/计算器
数据结构·c++·算法·1024程序员节
ZhiqianXia5 小时前
C++ 常见代码异味(Code Smells)
c++
编码追梦人6 小时前
基于 STM32 的智能语音唤醒与关键词识别系统设计 —— 从硬件集成到算法实现
stm32·算法·struts
循着风8 小时前
二叉树的多种遍历方式
数据结构·算法