D. Petya and Array(树状数组、动态开点)

简介: 笔记

题目描述


9.png


输入输出


10.png


样例


样例输入#1


5 4

5 -1 3 4 -1


样例输出#1


5


样例输入#2


3 0

-1 2 -3


样例输出#2


4


样例输入#3


4 -1

-2 1 -2 3


样例输出#3


3


数据范围


11.png


思路


枚举1−n 的每个数作为结尾,找到每个数前面前缀大于s[i]−t 的前缀有多少。

直接开一个前缀数组 s, 从小到大排序。

从前遍历,先将 s[i−1](即上一轮的pre),在排序的前缀和数组 s 中,二分出大于 pre 的下标 l,然后插入到树状数组中。

当前枚举的前缀和 pre(pre+=a[i];),在排序的前缀和数组s 中,二分出大于pre−t 的下标r。

大于这个 r 的即为答案做贡献。

这里需要注意,0 号位也参与。1<=l<=n,0<=l-1<=n-1

树状数组维护的是 i 前面有几个数小于等于它。

权值线段树

s[r] - s[l - 1] < t

s[l-1]>=s[r] - t + 1

s[r]<=s[l-1] + t - 1

与树状数组思路一致。

权值线段树维护值域,用于查询值落于区间[valL,valR] 的个数。

由于前缀和,所以最小 −2e14, 由于−t 所以值最小会在 −4e14。因为权值线段树维护的都是正值,所以将所有数加上 4e14,拉到正的范畴。

正向遍历


s[l−1]>=s[r]−t+1

将 s[i−1]插入树中,r即视为i,(前面小于i的位置都已插入树中),相当于询问前面 >=s[i]−t+1 的值的个数

for (int i = 1; i <= n; i++) {
    modify(root, L, R, s[i - 1], 1);
    res += query(root, L, R, s[i] - t + 1, R);
  }

反向遍历

s[r]<=s[l−1]+t−1

将s[i]插入树中,l即视为i,(后面大于i的位置都已插入树中),相当于询问后面<=s[i−1]+t−1

for (int i = n; i >= 1; i--) {
  modify(root, L, R, s[i], 1);
  res += query(root, L, R, 1, s[i - 1] + t - 1);
  }

代码


  • 树状数组


int n, t;
int tr[N], s[N], a[N];
int sum(int x) {
  int res = 0;
  for (; x; x -= x & -x) res += tr[x];
  return res;
}
void add(int x, int c) {
  for (; x <= n + 1; x += x & -x) tr[x] += c;
}
void solve() {
  cin >> n >> t;
  for (int i = 1; i <= n; i++) 
    cin >> a[i], s[i] = s[i - 1] + a[i];
  sort(s, s + 1 + n);
  int res = 0, pre = 0;
  for (int i = 1; i <= n; i++) {
    int l = upper_bound(s, s + 1 + n, pre) - s;
    add(l, 1);
    pre += a[i];      
    int r = upper_bound(s, s + 1 + n, pre - t) - s;
    res += sum(n + 1) - sum(r);
  }
  cout << res << endl;
}
  • 权值线段树
int n, t;
struct node {
  int l, r;
  int v;
} tr[N << 5];
int s[N], root, idx;
const int L = 1;
const int R = 1e15;
void pushup(int p) {
  tr[p].v = tr[tr[p].l].v + tr[tr[p].r].v;
}
void modify(int &p, int l, int r, int x, int v) {
  if(!p) p = ++idx;
  if(l == r) { tr[p].v += v; return ;}
  int mid = l + r >> 1;
  if(x <= mid) modify(tr[p].l, l, mid, x, v);
  if(x > mid) modify(tr[p].r, mid + 1, r, x, v);
  pushup(p);
}
int query(int p, int l, int r, int ql, int qr) {
  if(!p) return 0;
  if(l >= ql && r <= qr) return tr[p].v;
  int mid = l + r >> 1;
  int v = 0;
  if(ql <= mid) v = query(tr[p].l, l, mid, ql, qr);
  if(qr > mid) v += query(tr[p].r, mid + 1, r, ql, qr);
  return v;
}
void solve() {
  cin >> n >> t;
  for (int i = 1; i <= n; i++) {
    int x; cin >> x;
    s[i] = s[i - 1] + x;
  }
  for (int i = 0; i <= n; i++) s[i] += 4e14;
  int res = 0;
// s[r] - s[l - 1] < t
// s[l - 1] >= s[r] - t + 1
  // for (int i = 1; i <= n; i++) {
    // modify(root, L, R, s[i - 1], 1);
    // res += query(root, L, R, s[i] - t + 1, R);
  // }
// s[r] <= s[l - 1] + t - 1
  for (int i = n; i >= 1; i--) {
    modify(root, L, R, s[i], 1);
    res += query(root, L, R, 1, s[i - 1] + t - 1);
  }
  cout << res << endl;
}
相关文章
|
算法 Python
数组倍增(Array Doubling
数组倍增(Array Doubling)是一种常见的算法技术,用于解决数组相关的查找、插入、删除等问题。该技术的核心思想是将数组的大小乘以 2,新数组的长度是原数组长度的两倍,然后将原数组中的元素复制到新数组中。在某些情况下,这种技术可以提高算法的效率,尤其是对于动态数据结构的问题。
253 1
|
8月前
|
存储 算法 PHP
开发一款扑克游戏,请给出一套洗牌算法,公平的洗牌并将洗好的牌存储在一个整形数组里?
开发一款扑克游戏,请给出一套洗牌算法,公平的洗牌并将洗好的牌存储在一个整形数组里?
46 1
开发一款扑克游戏,请给出一套洗牌算法,公平的洗牌并将洗好的牌存储在一个整形数组里?
|
8月前
|
算法 测试技术 C#
【图论 状态压缩 枚举】2959. 关闭分部的可行集合数目
【图论 状态压缩 枚举】2959. 关闭分部的可行集合数目
|
8月前
7-7 念数字 (15 分)(用数组简化判断过程)
7-7 念数字 (15 分)(用数组简化判断过程)
58 0
重生之我是孔乙己——查找数组缺失元素的几种方法
重生之我是孔乙己——查找数组缺失元素的几种方法
86 0
|
存储 算法 前端开发
前端算法-除自身外数组的乘积
前端算法-除自身外数组的乘积
|
算法
(dfs)A -暴力模个拟(我是第一吗?我好像是第一个捏~)(原题目为Serval 的元素周期表)
(dfs)A -暴力模个拟(我是第一吗?我好像是第一个捏~)(原题目为Serval 的元素周期表)
63 0
枚举时对数组操——三刷AcWing 95. 费解的开关
枚举时对数组操——三刷AcWing 95. 费解的开关
70 0
|
算法 C++
<<算法很美>>——(四)——深入递归<二>——“逐步生成结果“类问题之非数值型
<<算法很美>>——(四)——深入递归<二>——“逐步生成结果“类问题之非数值型
<<算法很美>>——(四)——深入递归<二>——“逐步生成结果“类问题之非数值型
codeforces722——C.Destroying Array(并查集+栈+逆向思维)
codeforces722——C.Destroying Array(并查集+栈+逆向思维)
81 0
codeforces722——C.Destroying Array(并查集+栈+逆向思维)