Submission #73651999
Source Code Expand
#include <bits/stdc++.h>
#ifdef LOCAL
#include <library/my_library/utility/debug.hpp>
#else
#define debug(...)
#endif
using namespace std;
namespace rv = std::views; // NOLINT
// clang-format off
using lint = long long;
#define ALL(a) (a).begin(), (a).end()
template <class T> inline bool chmin(T &a, T b) { if (a > b) { a = b; return true; } return false; }
template <class T> inline bool chmax(T &a, T b) { if (a < b) { a = b; return true; } return false; }
inline void Yes(bool b = true) { println("{}",(b ? "Yes" : "No")); }
inline void No() { println("{}", "No");}
// clang-format on
#include <algorithm>
#include <cmath>
#include <functional>
#include <numeric>
#include <vector>
/**
* @brief Mo's Algorithm (平方分割によるオフラインクエリ処理)
* 使い方:
* Mo mo(N);
* mo.add_query(l, r);
* mo.solve(add_l, add_r, rem_l, rem_r, get_ans);
*/
struct Mo {
int n; // 配列のサイズ
int block_size; // ブロックサイズ (√N)
std::vector<std::pair<int, int>> queries; // クエリ (l, r)
std::vector<int> order; // クエリの処理順序
/**
* @brief コンストラクタ
* @param n 配列のサイズ
*/
Mo(int n) : n(n), block_size(std::sqrt(n) + 1) {}
/**
* @brief クエリを追加
* @param l 区間の左端 (0-indexed)
* @param r 区間の右端 (0-indexed, 半開区間 [l, r))
*/
void add_query(int l, int r) { queries.emplace_back(l, r); }
/**
* @brief クエリを最適な順序でソート
* @note 左端のブロック番号でソート、同じブロックなら右端でソート
* 奇数ブロックは右端を降順にすることで移動回数を削減
*/
void build() {
int q = queries.size();
order.resize(q);
std::iota(order.begin(), order.end(), 0);
std::sort(order.begin(), order.end(), [&](int i, int j) {
int block_i = queries[i].first / block_size;
int block_j = queries[j].first / block_size;
if (block_i != block_j) return block_i < block_j;
// 奇数ブロックは右端を降順にする (Hilbert曲線の近似)
if (block_i & 1) return queries[i].second > queries[j].second;
return queries[i].second < queries[j].second;
});
}
/**
* @brief クエリを処理して結果を返す
*
* @tparam AddLeft 左端を追加する関数 (int idx) -> void
* @tparam AddRight 右端を追加する関数 (int idx) -> void
* @tparam RemoveLeft 左端を削除する関数 (int idx) -> void
* @tparam RemoveRight 右端を削除する関数 (int idx) -> void
* @tparam GetAnswer 現在の状態から答えを取得する関数 () -> T
*
* @return クエリの答えの配列 (元のクエリの順序)
*/
template <typename AddLeft, typename AddRight, typename RemoveLeft, typename RemoveRight, typename GetAnswer>
auto solve(AddLeft add_left, AddRight add_right, RemoveLeft remove_left, RemoveRight remove_right,
GetAnswer get_answer) {
using Answer = decltype(get_answer());
int q = queries.size();
std::vector<Answer> answers(q);
build();
int current_l = 0, current_r = 0;
for (int idx : order) {
auto [l, r] = queries[idx];
// 左端を拡張
while (current_l > l) {
current_l--;
add_left(current_l);
}
// 右端を拡張
while (current_r < r) {
add_right(current_r);
current_r++;
}
// 左端を縮小
while (current_l < l) {
remove_left(current_l);
current_l++;
}
// 右端を縮小
while (current_r > r) {
current_r--;
remove_right(current_r);
}
answers[idx] = get_answer();
}
return answers;
}
/**
* @brief クエリを処理して結果を返す (簡易版)
* @note add/remove が左右で同じ処理の場合に使用
*
* @tparam Add 要素を追加する関数 (int idx) -> void
* @tparam Remove 要素を削除する関数 (int idx) -> void
* @tparam GetAnswer 現在の状態から答えを取得する関数 () -> T
*
* @return クエリの答えの配列 (元のクエリの順序)
*/
template <typename Add, typename Remove, typename GetAnswer>
auto solve(Add add, Remove remove, GetAnswer get_answer) {
return solve(add, add, remove, remove, get_answer);
}
};
int main() {
cin.tie(nullptr);
ios::sync_with_stdio(false);
int N, Q;
cin >> N >> Q;
Mo mo(N);
vector<int> A(N);
for (auto&& a : A) cin >> a;
for (int i = 0; i < Q; i++) {
int l, r;
cin >> l >> r;
l--;
mo.add_query(l, r);
}
mo.build();
unordered_map<int, int> mp;
auto add = [&](int r) { mp[A[r]]++; };
auto remove = [&](int l) {
mp[A[l]]--;
if (mp[A[l]] == 0) mp.erase(A[l]);
};
auto get_ans = [&] { return ssize(mp); };
auto ans = mo.solve(add, add, remove, remove, get_ans);
for (int i = 0; i < ssize(ans); i++) {
cout << ans[i] << '\n';
}
}
Submission Info
| Submission Time | |
|---|---|
| Task | E - Count the Types of Flowers |
| User | roromiya_tenkei |
| Language | C++23 (GCC 15.2.0) |
| Score | 466 |
| Code Size | 5621 Byte |
| Status | AC |
| Exec Time | 646 ms |
| Memory | 10376 KiB |
Judge Result
| Set Name | Sample | All | ||||
|---|---|---|---|---|---|---|
| Score / Max Score | 0 / 0 | 466 / 466 | ||||
| Status |
|
|
| Set Name | Test Cases |
|---|---|
| Sample | sample01.txt, sample02.txt, sample03.txt |
| All | sample01.txt, sample02.txt, sample03.txt, in01.txt, in02.txt, in03.txt, in04.txt, in05.txt, in06.txt, in07.txt, in08.txt, in09.txt, in10.txt, in11.txt, in12.txt, in13.txt, in14.txt, in15.txt, in16.txt, in17.txt, in18.txt, in19.txt, in20.txt, in21.txt, in22.txt, in23.txt, in24.txt, in25.txt, in26.txt, in27.txt, in28.txt, in29.txt, in30.txt, in31.txt, in32.txt, in33.txt, in34.txt, in35.txt, in36.txt, in37.txt, in38.txt, in39.txt, in40.txt, in41.txt, in42.txt, in43.txt, in44.txt, in45.txt, in46.txt, in47.txt, in48.txt, in49.txt, in50.txt, in51.txt, in52.txt, in53.txt, in54.txt, in55.txt, in56.txt, in57.txt, in58.txt, in59.txt, in60.txt, in61.txt, in62.txt, in63.txt, in64.txt, in65.txt, in66.txt, in67.txt, in68.txt, in69.txt, in70.txt, in71.txt, in72.txt, in73.txt, in74.txt |
| Case Name | Status | Exec Time | Memory |
|---|---|---|---|
| in01.txt | AC | 1 ms | 3580 KiB |
| in02.txt | AC | 1 ms | 3676 KiB |
| in03.txt | AC | 1 ms | 3616 KiB |
| in04.txt | AC | 1 ms | 3560 KiB |
| in05.txt | AC | 1 ms | 3408 KiB |
| in06.txt | AC | 1 ms | 3600 KiB |
| in07.txt | AC | 1 ms | 3524 KiB |
| in08.txt | AC | 1 ms | 3544 KiB |
| in09.txt | AC | 646 ms | 8256 KiB |
| in10.txt | AC | 161 ms | 5740 KiB |
| in11.txt | AC | 638 ms | 8248 KiB |
| in12.txt | AC | 157 ms | 5820 KiB |
| in13.txt | AC | 8 ms | 6384 KiB |
| in14.txt | AC | 11 ms | 8424 KiB |
| in15.txt | AC | 4 ms | 3668 KiB |
| in16.txt | AC | 429 ms | 7276 KiB |
| in17.txt | AC | 24 ms | 10300 KiB |
| in18.txt | AC | 43 ms | 5688 KiB |
| in19.txt | AC | 143 ms | 5688 KiB |
| in20.txt | AC | 142 ms | 5684 KiB |
| in21.txt | AC | 532 ms | 10376 KiB |
| in22.txt | AC | 540 ms | 10208 KiB |
| in23.txt | AC | 142 ms | 5816 KiB |
| in24.txt | AC | 210 ms | 8260 KiB |
| in25.txt | AC | 24 ms | 8260 KiB |
| in26.txt | AC | 399 ms | 8276 KiB |
| in27.txt | AC | 15 ms | 5260 KiB |
| in28.txt | AC | 1 ms | 3484 KiB |
| in29.txt | AC | 1 ms | 3480 KiB |
| in30.txt | AC | 1 ms | 3688 KiB |
| in31.txt | AC | 1 ms | 3480 KiB |
| in32.txt | AC | 93 ms | 8388 KiB |
| in33.txt | AC | 66 ms | 7668 KiB |
| in34.txt | AC | 1 ms | 3616 KiB |
| in35.txt | AC | 1 ms | 3568 KiB |
| in36.txt | AC | 1 ms | 3728 KiB |
| in37.txt | AC | 1 ms | 3528 KiB |
| in38.txt | AC | 1 ms | 3616 KiB |
| in39.txt | AC | 1 ms | 3564 KiB |
| in40.txt | AC | 1 ms | 3448 KiB |
| in41.txt | AC | 1 ms | 3804 KiB |
| in42.txt | AC | 3 ms | 3844 KiB |
| in43.txt | AC | 103 ms | 5700 KiB |
| in44.txt | AC | 104 ms | 5688 KiB |
| in45.txt | AC | 76 ms | 8272 KiB |
| in46.txt | AC | 27 ms | 8268 KiB |
| in47.txt | AC | 35 ms | 8244 KiB |
| in48.txt | AC | 378 ms | 7988 KiB |
| in49.txt | AC | 277 ms | 8428 KiB |
| in50.txt | AC | 173 ms | 8240 KiB |
| in51.txt | AC | 82 ms | 5620 KiB |
| in52.txt | AC | 577 ms | 8268 KiB |
| in53.txt | AC | 84 ms | 5632 KiB |
| in54.txt | AC | 567 ms | 8316 KiB |
| in55.txt | AC | 160 ms | 5844 KiB |
| in56.txt | AC | 215 ms | 5836 KiB |
| in57.txt | AC | 636 ms | 8256 KiB |
| in58.txt | AC | 619 ms | 8260 KiB |
| in59.txt | AC | 1 ms | 3600 KiB |
| in60.txt | AC | 1 ms | 3616 KiB |
| in61.txt | AC | 167 ms | 5812 KiB |
| in62.txt | AC | 167 ms | 5732 KiB |
| in63.txt | AC | 1 ms | 3516 KiB |
| in64.txt | AC | 1 ms | 3564 KiB |
| in65.txt | AC | 1 ms | 3580 KiB |
| in66.txt | AC | 1 ms | 3616 KiB |
| in67.txt | AC | 1 ms | 3548 KiB |
| in68.txt | AC | 1 ms | 3688 KiB |
| in69.txt | AC | 1 ms | 3448 KiB |
| in70.txt | AC | 1 ms | 3568 KiB |
| in71.txt | AC | 3 ms | 3716 KiB |
| in72.txt | AC | 3 ms | 3772 KiB |
| in73.txt | AC | 474 ms | 8268 KiB |
| in74.txt | AC | 392 ms | 8272 KiB |
| sample01.txt | AC | 1 ms | 3408 KiB |
| sample02.txt | AC | 1 ms | 3544 KiB |
| sample03.txt | AC | 1 ms | 3588 KiB |