gpt4 book ai didi

algorithm - (ACM) 如何用线段树统计[a,b]中有多少元素小于给定常数?

转载 作者:塔克拉玛干 更新时间:2023-11-03 03:12:01 25 4
gpt4 key购买 nike

我对线段树还很陌生,想多做一些关于线段树的练习让自己忙起来。

这个问题实际上更像 ACM,并且具有以下条件:有n个数,m个操作,n,m<=10,000,每个操作可以是以下之一: 1.通过减去一个数x来更新一个区间,每次x可以不同 2.查询一个区间,找出区间中有多少个数是<=0

在这里构建线段树和更新显然可以在 O(nlog n)/O(log n) 中完成但是我不知道如何在 O(log n) 中进行查询,谁能给我一些建议/提示?任何的意见都将会有帮助!谢谢!

长话短说:

给定 n 个数字和 2 个类型的操作:

  1. 将x添加到[a,b]中的所有元素,每次x可以不同
  2. 查询[a,b]中元素个数为

如何使操作 1 和 2 都可以在 O(log n) 中完成?

最佳答案

好问题:)

想了想还是没能解决这个线段树的问题,不过我试过用“桶法”来解决这个问题。

我们可以将最初的n个数分成B个桶,对每个桶中的数字进行排序,并维护每个桶中的总加值。然后对于每个查询:

  • 用c“添加”更新间隔[a, b]

    我们最多只需要重建两个桶,并将 c 添加到 (b - a)/BUCKET_SIZE 个桶中

  • "查询"查询区间[a, b] <= c

    我们只需要一个一个地扫描最多两个桶,每个桶都有一个值,然后用二分查找快速遍历 (b-a)/BUCKET_SIZE 个桶

对于每个查询,它应该在 O( N/BUCKET_SIZE * log(BUCKET_SIZE, 2)) 内运行,这比暴力法( O(N) )小。虽然它比 O(logN) 大,但在大多数情况下可能就足够了。

测试代码如下:

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <string>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <vector>
#include <set>
#include <map>
#include <ctime>
#include <cassert>

using namespace std;

struct Query {
//A a b c add c in [a, b] of arr
//Q a b c Query number of i in [a, b] which arr[i] <= c
char ty;
int a, b, c;
Query(char _ty, int _a, int _b, int _c):ty(_ty), a(_a), b(_b), c(_c){}
};

int n, m;
vector<int> arr;
vector<Query> queries;

vector<int> bruteforce() {
vector<int> ret;
vector<int> numbers = arr;
for (int i = 0; i < m; i++) {
Query q = queries[i];
if (q.ty == 'A') {
for (int i = q.a; i <= q.b; i++) {
numbers[i] += q.c;
}
ret.push_back(-1);
} else {
int tmp = 0;
for(int i = q.a; i <= q.b; i++) {
tmp += numbers[i] <= q.c;
}
ret.push_back(tmp);
}
}
return ret;
}

struct Bucket {
vector<int> numbers;
vector<int> numbers_sorted;
int add;
Bucket() {
add = 0;
numbers_sorted.clear();
numbers.clear();
}
int query(int pos) {
return numbers[pos] + add;
}
void add_pos(int pos, int val) {
numbers[pos] += val;
}
void build() {
numbers_sorted = numbers;
sort(numbers_sorted.begin(), numbers_sorted.end());
}
};

vector<int> bucket_count(int bucket_size) {
vector<int> ret;

vector<Bucket> buckets;
buckets.resize(int(n / bucket_size) + 5);
for (int i = 0; i < n; i++) {
buckets[i / bucket_size].numbers.push_back(arr[i]);
}

for (int i = 0; i <= n / bucket_size; i++) {
buckets[i].build();
}

for (int i = 0; i < m; i++) {
Query q = queries[i];
char ty = q.ty;
int a, b, c;
a = q.a, b = q.b, c = q.c;
if (ty == 'A') {
set<int> affect_buckets;
while (a < b && a % bucket_size != 0) buckets[a/ bucket_size].add_pos(a % bucket_size, c), affect_buckets.insert(a/bucket_size), a++;
while (a < b && b % bucket_size != 0) buckets[b/ bucket_size].add_pos(b % bucket_size, c), affect_buckets.insert(b/bucket_size), b--;
while (a < b) {
buckets[a/bucket_size].add += c;
a += bucket_size;
}
buckets[a/bucket_size].add_pos(a % bucket_size, c), affect_buckets.insert(a / bucket_size);
for (set<int>::iterator it = affect_buckets.begin(); it != affect_buckets.end(); it++) {
int id = *it;
buckets[id].build();
}
ret.push_back(-1);
} else {
int tmp = 0;
while (a < b && a % bucket_size != 0) tmp += (buckets[a/ bucket_size].query(a % bucket_size) <=c), a++;
while (a < b && b % bucket_size != 0) tmp += (buckets[b/ bucket_size].query(b % bucket_size) <=c), b--;
while (a < b) {
int pos = a / bucket_size;
tmp += upper_bound(buckets[pos].numbers_sorted.begin(), buckets[pos].numbers_sorted.end(), c - buckets[pos].add) - buckets[pos].numbers_sorted.begin();
a += bucket_size;
}
tmp += (buckets[a / bucket_size].query(a % bucket_size) <= c);
ret.push_back(tmp);
}
}

return ret;
}

void process(int cas) {

clock_t begin_t=clock();

vector<int> bf_ans = bruteforce();
clock_t bf_end_t =clock();
double bf_sec = ((1.0 * bf_end_t - begin_t)) / CLOCKS_PER_SEC;

//bucket_size is important
int bucket_size = 200;
vector<int> ans = bucket_count(bucket_size);

clock_t bucket_end_t =clock();
double bucket_sec = ((1.0 * bucket_end_t - bf_end_t)) / CLOCKS_PER_SEC;

bool correct = true;
for (int i = 0; i < ans.size(); i++) {
if (ans[i] != bf_ans[i]) {
cout << "query " << i + 1 << " bf = " << bf_ans[i] << " bucket = " << ans[i] << " bucket size = " << bucket_size << " " << n << " " << m << endl;
correct = false;
}
}
printf("Case #%d:%s bf_sec = %.9lf, bucket_sec = %.9lf\n", cas, correct ? "YES":"NO", bf_sec, bucket_sec);
}

void read() {
cin >> n >> m;
arr.clear();
for (int i = 0; i < n; i++) {
int val;
cin >> val;
arr.push_back(val);
}
queries.clear();
for (int i = 0; i < m; i++) {
char ty;
int a, b, c;
// a, b, c in [0, n - 1], a <= b
cin >> ty >> a >> b >> c;
queries.push_back(Query(ty, a, b, c));
}
}

void run(int cas) {
read();
process(cas);
}

int main() {
freopen("bucket.in", "r", stdin);
//freopen("bucket.out", "w", stdout);
int T;
scanf("%d", &T);
for (int cas = 1; cas <= T; cas++) {
run(cas);
}
return 0;
}

这里是数据生成代码:

#coding=utf8

import random
import math

def gen_buckets(f):
t = random.randint(10, 20)
print >> f, t
nlimit = 100000
mlimit = 10000
limit = 100000
for i in xrange(t):
n = random.randint(1, nlimit)
m = random.randint(1, mlimit)
print >> f, n, m

for i in xrange(n):
val = random.randint(1, limit)
print >> f, val ,
print >> f
for i in xrange(m):
ty = random.randint(1, 2)
a = random.randint(0, n - 1)
b = random.randint(a, n - 1)
#a = 0
#b = n - 1
c = random.randint(-limit, limit)
print >> f, 'A' if ty == 1 else 'Q', a, b, c


f = open("bucket.in", "w")
gen_buckets(f)

关于algorithm - (ACM) 如何用线段树统计[a,b]中有多少元素小于给定常数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18800058/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com