У меня массивы, before[N+1]
(индексируется) и after[]
(подмассива before[]
). Теперь для M Запросов, мне нужно найти, сколько элементов after[]
присутствует в before[]
для данного диапазона l,r
.MO, чтобы найти число элементов, присутствующих в обоих массиве
Например:
N = 5
Перед: (2, 1, 3, 4, 5)
После того, как: (1, 3, 4, 5)
М = 2
L = 1 , R = 5 → 4 элемента (1, 3, 4, 5) после [] присутствуют между до [1] и до [5]
L = 2, R = 4 → 3 элемента (1, 3, 4) of after [] присутствуют между до [2] и до [4]
Я пытаюсь использовать алгоритм MO, чтобы найти это. Следующим является мой код:
using namespace std;
int N, Q;
// Variables, that hold current "state" of computation
long long current_answer;
long long cnt[100500];
// Array to store answers (because the order we achieve them is messed up)
long long answers[100500];
int BLOCK_SIZE;
// We will represent each query as three numbers: L, R, idx. Idx is
// the position (in original order) of this query.
pair< pair<int, int>, int> queries[100500];
// Essential part of Mo's algorithm: comparator, which we will
// use with std::sort. It is a function, which must return True
// if query x must come earlier than query y, and False otherwise.
inline bool mo_cmp(const pair< pair<int, int>, int> &x,
const pair< pair<int, int>, int> &y)
{
int block_x = x.first.first/BLOCK_SIZE;
int block_y = y.first.first/BLOCK_SIZE;
if(block_x != block_y)
return block_x < block_y;
return x.first.second < y.first.second;
}
// When adding a number, we first nullify it's effect on current
// answer, then update cnt array, then account for it's effect again.
inline void add(int x)
{
current_answer -= cnt[x] * cnt[x] * x;
cnt[x]++;
current_answer += cnt[x] * cnt[x] * x;
}
// Removing is much like adding.
inline void remove(int x)
{
current_answer -= cnt[x] * cnt[x] * x;
cnt[x]--;
current_answer += cnt[x] * cnt[x] * x;
}
int main()
{
cin.sync_with_stdio(false);
cin >> N >> Q; // Q- number of queries
BLOCK_SIZE = static_cast<int>(sqrt(N));
long long int before[N+1]; // 1 indexed
long long int after[] // subarray
// Read input queries, which are 0-indexed. Store each query's
// original position. We will use it when printing answer.
for(long long int i = 0; i < Q; i++) {
cin >> queries[i].first.first >> queries[i].first.second;
queries[i].second = i;
}
// Sort queries using Mo's special comparator we defined.
sort(queries, queries + Q, mo_cmp);
// Set up current segment [mo_left, mo_right].
int mo_left = 0, mo_right = -1;
for(long long int i = 0; i < Q; i++) {
// [left, right] is what query we must answer now.
int left = queries[i].first.first;
int right = queries[i].first.second;
// Usual part of applying Mo's algorithm: moving mo_left
// and mo_right.
while(mo_right < right) {
mo_right++;
add(after[mo_right]);
}
while(mo_right > right) {
remove(after[mo_right]);
mo_right--;
}
while(mo_left < left) {
remove(after[mo_left]);
mo_left++;
}
while(mo_left > left) {
mo_left--;
add(after[mo_left]);
}
// Store the answer into required position.
answers[queries[i].second] = current_answer;
}
// We output answers *after* we process all queries.
for(long long int i = 0; i < Q; i++)
cout << answers[i] << "\n";
Теперь проблема, я не могу понять, как определить add function
и remove function
.
Может ли кто-нибудь помочь мне с этими функциями?
Являются ли элементы до [] уникальными? Если да, почему бы не сделать отметку boolean array [], в которой знак [i] представляет, появляется ли перед [i] после []? Тогда запрос (l, r) становится считанным числом истины между отметкой [l] и меткой [r]. –
Если 'after []' is *** subarray *** 'before []', то 'NumberOfOccurrence (NOC)' никогда не может быть больше, чем 'NumberOfElements (An)' in 'after []'. 'NOC = max (r-l + 1, An)' – sameerkn
@MoTao Нет, элементы в 'before []' не уникальны. – Buckster