結果
問題 | No.3078 Difference Sum Query |
ユーザー |
![]() |
提出日時 | 2025-03-28 23:40:32 |
言語 | C++23 (gcc 13.3.0 + boost 1.87.0) |
結果 |
WA
|
実行時間 | - |
コード長 | 6,798 bytes |
コンパイル時間 | 4,536 ms |
コンパイル使用メモリ | 312,080 KB |
実行使用メモリ | 39,128 KB |
最終ジャッジ日時 | 2025-03-28 23:40:59 |
合計ジャッジ時間 | 25,981 ms |
ジャッジサーバーID (参考情報) |
judge3 / judge2 |
(要ログイン)
ファイルパターン | 結果 |
---|---|
sample | AC * 2 |
other | AC * 14 WA * 12 |
ソースコード
#include <bits/stdc++.h>//#include <atcoder/modint>using namespace std;//using namespace atcoder;using ll = long long;//using mint = modint998244353;// Bit Vector (for 64-bit non-negative integer)struct BitVector {// block: bit vector// count: the number of 1 within each blockunsigned int n, zeros;vector<unsigned long long> block;vector<unsigned int> count;// constructorBitVector() {}BitVector(const unsigned int num) {resize(num);}void resize(const unsigned int num) {n = num;block.assign(((num + 1) >> 6) + 1, 0);count.assign(block.size(), 0);}// set val(0 or 1) onto i-th bit, get i-th bit of val(0 or 1)void set(const unsigned int i, const unsigned long long val = 1LL) {assert((i >> 6) < block.size());block[i >> 6] |= (val << (i & 63));}unsigned int get(const unsigned int i) const {assert((i >> 6) < block.size());return (const unsigned int)(block[i >> 6] >> (i & 63)) & 1;}void build() {for (unsigned int i = 1; i < block.size(); i++) {count[i] = count[i - 1] + __builtin_popcountll(block[i - 1]);}zeros = rank0(n);}// the number of 1 in [0, i)unsigned int rank1(const unsigned int i) const {assert((i >> 6) < count.size());assert((i >> 6) < block.size());return count[i >> 6] +__builtin_popcountll(block[i >> 6] & ((1ULL << (i & 63)) - 1ULL));}// the number of 1 in [i, j)unsigned int rank1(const unsigned int i, const unsigned int j) const {return rank1(j) - rank1(i);}// the number of 0 in [0, i)unsigned int rank0(const unsigned int i) const {return i - rank1(i);}// the number of 0 in [i, j)unsigned int rank0(const unsigned int i, const unsigned int j) const {return rank0(j) - rank0(i);}// the number of 0 in [0, n)unsigned int rank0() const {return zeros;}};// BIT on Wavelet Matrixtemplate<class POS, class VAL> struct BITonWaveletMatrix {// inner datastruct BIT {VAL UNITY_SUM = 0;int N;vector<VAL> dat;// [0, n)BIT() {}BIT(int n, VAL unity = 0) : UNITY_SUM(unity), N(n), dat(n, unity) { }void init(int n) {N = n;dat.assign(n, UNITY_SUM);}// a is 0-indexedvoid add(int a, VAL x) {for (int i = a; i < (int)dat.size(); i |= i + 1)dat[i] = dat[i] + x;}// [0, a), a is 0-indexedVAL sum(int a) {VAL res = UNITY_SUM;for (int i = a - 1; i >= 0; i = (i & (i + 1)) - 1)res = res + dat[i];return res;}// [a, b), a and b are 0-indexedVAL sum(int a, int b) {return sum(b) - sum(a);}// debugvoid print() {for (int i = 0; i < (int)dat.size(); ++i)cout << sum(i, i + 1) << ",";cout << endl;}};using Point = pair<POS, POS>;int n, height;vector<BitVector> bv;vector<Point> ps;vector<POS> ys;vector<BIT> bit;// constructor (sigma: the number of characters)// add_point() cannot be used after build()BITonWaveletMatrix() {}BITonWaveletMatrix(const vector<Point> &vec) {for (auto [x, y] : vec) add_point(x, y);build();}void add_point(POS x, POS y) {ps.emplace_back(x, y);ys.emplace_back(y);}int xid(POS x) const {return lower_bound(ps.begin(), ps.end(), Point(x, 0)) - ps.begin();}int yid(POS y) const {return lower_bound(ys.begin(), ys.end(), y) - ys.begin();}void build() {sort(ps.begin(), ps.end());ps.erase(unique(ps.begin(), ps.end()), ps.end());n = (int)ps.size();sort(ys.begin(), ys.end());ys.erase(unique(ys.begin(), ys.end()), ys.end());vector<int> v(n), left(n), right(n), ord(n);int mv = 1;for (int i = 0; i < n; ++i) {v[i] = yid(ps[i].second);mv = max(mv, v[i]);}for (height = 1; mv != 0; mv >>= 1) ++height;iota(ord.begin(), ord.end(), 0);bv.assign(height, BitVector(n));bit.assign(height + 1, BIT(n));for (int h = height - 1; h >= 0; --h) {int l = 0, r = 0;for (int i = 0; i < n; ++i) {if ((v[ord[i]] >> h) & 1) {bv[h].set(i);right[r++] = ord[i];} else {left[l++] = ord[i];}}bv[h].build();ord.swap(left);for (int i = 0; i < r; ++i) ord[i + l] = right[i];}}// addvoid add(const POS x, const POS y, const VAL val) {int i = lower_bound(ps.begin(), ps.end(), Point(x, y)) - ps.begin();int j = yid(y);for (int h = height - 1; h >= 0; --h) {int i0 = bv[h].rank0(i);if ((j >> h) & 1) {i += bv[h].rank0() - i0;} else {i = i0;}bit[h].add(i, val);}}// sumVAL inner_sum(int l, int r, const POS upper) {assert(0 <= l && r <= n);VAL res = 0;for (int h = height - 1; h >= 0; --h) {int l0 = bv[h].rank0(l), r0 = bv[h].rank0(r);if ((upper >> h) & 1) {l += bv[h].rank0() - l0;r += bv[h].rank0() - r0;res += bit[h].sum(l0, r0);} else {l = l0;r = r0;}}return res;}VAL sum(const POS lx, const POS rx, const POS ly, const POS ry) {int l = xid(lx), r = xid(rx);return inner_sum(l, r, yid(ry)) - inner_sum(l, r, yid(ly));}};int main() {cin.tie(nullptr);ios_base::sync_with_stdio(false);BITonWaveletMatrix<int, ll> wms, wmc;ll N, Q, M=1e10;cin >> N >> Q;vector<ll> A(N);for (int i = 0; i < N; i++) {cin >> A[i];wms.add_point(i, A[i]);wmc.add_point(i, A[i]);}wms.build();wmc.build();for (int i = 0; i < N; i++) {wms.add(i, A[i], A[i]);wmc.add(i, A[i], 1);}while (Q--) {ll L, R, X;cin >> L >> R >> X;L--; R--;ll c1 = wmc.sum(L, R + 1, 0, X);ll c2 = wmc.sum(L, R + 1, X, M+1);ll s1 = wms.sum(L, R + 1, 0, X);ll s2 = wms.sum(L, R + 1, X, M+1);cout << c1 * X - s1 + s2 - c2 * X << '\n';}return 0;}