ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

常见的数据结构梳理

2021-09-14 20:33:22  阅读:176  来源: 互联网

标签:const int sum 常见 mid pos 数组 数据结构 梳理


线段树

复习了一会线段树, 觉得线段树最精妙的地方就在于 lazyTag 表达的含义

lazyTag 打的标记是该节点的所有子节点, 不包括自己

分析更新的正确性, 对于当前节点, 执行完 push_down 操作后, 可以保证其左右孩子上的 sum 值都是正确的

所以最后一句 sum[o] = sum[lo] + sum[ro] 的正确性可以保障

void updt(const int& L,const int& R,const int64_t& val, int o = 1,int l = 1,int r = n){
    if(L <= l and r <= R){
        sum[o] += (r - l + 1) * val;
        lz[o] += val;
        return;
    }
    push_down(o, l, r);
    if(L <= mid) updt(L, R, val, lo, l, mid);
    if(R > mid) updt(L, R, val, ro, mid + 1, r);
    sum[o] = sum[lo] + sum[ro];
}

考虑进行更新时经过的节点, 对于每一层, 最多只会经过两个节点, 因此复杂度是 \(log n\)

树状数组

树状数组的英文名是 Binary Indexed Trees

image-20210914163339501

对于 C[i] , 它维护的是一个以 i 结尾, 长度为 lowbit(i) 的区间和, 如上图所示

查询 [1, x] 的区间和, 其实就是对 x 进行了一个从后往前的二进制拆分, 因为维护的是类似后缀的东西,

显然 S[1,x] = C[x] + C[x - lowbit(x)] , 不停的跳即可

对于一个单点的更新操作, 每次向上跳 lowbit , 可以保证是最小且有效的步数

void add(int pos,int val){
    for(;pos <= n;pos += -pos & pos) C[pos] += val;
}
int query(int p){
    int ans = 0;
    for(;p;p -= -pos & pos) ans += C[p];
    return ans;
}

树状数组也可以用来区间加减,区间查询

设 \(A_i\) 是原数组, \(d_i = A_i - A_{i-1}\) 是 \(A\) 的差分数组

\[S_n = \sum_{i=1}^nA_i = d_1 + \\d_1 + d_2 +\\ d_1 + d_2 + d_3 + \\ ... \\d_1 + d_2 + d_3 + d_4 + ... + d_n \\ = n (d_1 + d_2 + ... + d_n) - \sum_{i=1}^nd_i(i-1) \]

设 \(B_i = d_i(i-1)\)

\[S_n = n\sum d - \sum B \]

而当一段区间 \([L,R]\) 一起加上 \(x\) 的时候,

对于 \(d\) 来说,只有 \(d_L\) 和 \(d_{R+1}\) 发生了变化

对于 \(B\) 来说,也是同理,所以可以开两个树状数组来进行维护

struct{
	  ll C[N][2]; // 0 是差分d_i , 1 是 d_i * (i - 1)
	  void add(int pos, ll val, int o) {
	  	  for (; pos <= n; pos += (-pos) & pos) C[pos][o] += val;
	  }
	  ll ask(int pos, int o) {
	  	  ll ans = 0;
	  	  for (; pos; pos -= (-pos) & pos) ans += C[pos][o];
	  	  return ans;
	  }

	  void updt(int l, int r, int x) {
	  	  add(l, x, 0); add(r + 1, -x, 0);
	  	  add(l, x * (l - 1), 1); add(r + 1, -x * (r), 1);
	  }
	  ll query(int l, int r) {
	  	  ll R = r * ask(r, 0) - ask(r, 1);
	  	  ll L = (l - 1) * ask(l - 1, 0) - ask(l - 1, 1);
	  	  return R - L;
	  }
}BIT;

主席树

其实就是一个桶, 挂链的时候复制链的信息动态开点,查询区间 k 大的时候,对两棵树进行查询即可

挂链可以挂成线性的可以挂成树形

#include<bits/stdc++.h>
#define mid (l+r>>1)
using namespace std;

const int maxn = 5e5 + 10;

int sum[maxn << 5], L[maxn << 5], R[maxn << 5];
int cnt;

int a[maxn], id[maxn], root[maxn];

int build(int l, int r) {
    int rt = ++cnt;
    sum[rt] = 0;
    if (l < r) {
        L[rt] = build(l, mid);
        R[rt] = build(mid + 1, r);
    }
    return rt;
}

int updt(int pre, int l, int r, int pos) {
    int rt = ++cnt;
    sum[rt] = sum[pre] + 1;

    R[rt] = R[pre];
    L[rt] = L[pre];

    if (l < r) {
        if (pos <= mid) {
            L[rt] = updt(L[pre], l, mid, pos);
        }
        else {
            R[rt] = updt(R[pre], mid + 1, r, pos);
        }
    }
    return rt;
}

int queryK(int x, int y, int l, int r, int k) {
    if (l == r) {
        return r;
    }
    int num = sum[L[y]] - sum[L[x]];
    if (num >= k) {
        return queryK(L[x], L[y], l, mid, k);
    }
    else {
        return queryK(R[x], R[y], mid + 1, r, k - num);
    }
}


标签:const,int,sum,常见,mid,pos,数组,数据结构,梳理
来源: https://www.cnblogs.com/sduwh/p/15269469.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有