逆序对2.0
我们称逆序对为一个序列中满足 且 的二元组 。
若一个排列的逆序对个数为奇数,则称它为一个奇排列,否则它被称为偶排列。
给出一个长度为 的排列,有 种操作:
- 交换 和 。
- 翻转区间 内的数。
- 将区间 内的数左移 位。
- 将区间 内的数右移 位。
对一个长度为 n 的序列的操作定义如下:
- 对于所有 ,将第 个位置的数移动到第 个位置。
- 对于所有 ,将第 个位置的数移动到第 个位置,第 个位置的数移动到第 个位置。重复此操作 次。
- 对于所有 ,将第 个位置的数移动到第 个位置,第 个位置的数移动到第 个位置。重复此操作 次。
注意上述定义中同一个操作内的移动都是同时进行的。
想要知道每次操作之后这个排列是奇排列还是偶排列。
第一行两个整数 ,表示序列的长度和操作的次数。
第二行 个整数,表示一个 至 的排列。
接下来 行,每行三或四个整数,表示一次操作,具体格式及其含义见题目描述。
共 行,表示每次操作后的答案,若为奇排列,则输出 , 若为偶排列则输出 。
样例一:
5 5
1 2 3 4 5
1 1 3
1 2 3
2 2 5
3 1 2 1
4 3 5 1
样例二:
10 10
1 4 3 5 2 9 8 6 7 10
1 2 3
2 3 5
4 3 7 2
2 3 5
3 5 7 4
1 2 6
4 1 10 6
3 2 4 3
1 3 4
2 3 9
样例一:
1
0
0
1
1
样例二:
0
1
1
0
0
1
1
1
0
1
样例一说明:
原排列的逆序对个数为 。
第一次操作后序列为: ,逆序对个数为 ,故输出 。
第二次操作后序列为: ,逆序对个数为 ,故输出 。
第三次操作后序列为: ,逆序对个数为 ,故输出 。
第四次操作后序列为: ,逆序对个数为 9,故输出 。
第五次操作后序列为: ,逆序对个数为 7,故输出 。
数据范围:
对于前 的数据,。
对于前 的数据,。
对于前 的数据,。
对于 的数据,。
证明
先研究一下相邻两个元素的对换
考虑序列
对换后
我们可以知道, 与 对换后,并不影响 对 , 对 , 对 , 对 的大小顺序排列,只有 与 之间的大小排列发生了改变,假设 , 之间的排列为顺序,则对换后,变为逆序,逆序数加 ,改变奇偶性,若 , 之间的排列为逆序,则对换后,变为顺序,逆序数减 , 改变奇偶性
现在我们来考虑一下一般情况
经过 共 次对换后得到
在经 共 次对换后得到
又经过 共 次对换后得到
合计共经过 次对换,显然 为奇数
由引理 可以知道,奇数次对换改变排列奇偶性
证毕
解题步骤如下:
先求出原始序列的逆序对数的奇偶性(可以通过归并排序或者树状数组的方法求逆序对数)
操作1. 交换 次( 与 不相等) 操作2. 对半交换,相当于 次操作 操作3. 依次向左(或向右)逐位交换。此时操作的序列长度为 ,那么相当于做 次操作
这里拓展一个知识点:假设整个序列向右移 位,相当于将前 位翻转,对后 位翻转,最后对整个序列翻转。
执行上述 步即可,这 步的顺序可以任意,这样处理的优势是时间复杂度控制在 ,并且不需要另开空间,直接在原数组上就可以完成。
#include <bits/stdc++.h>
#define len b - a + 1
using namespace std;
inline int read()
{
char c = getchar();
int x = 0;
bool f = 0;
for (; !isdigit(c); c = getchar())
{
f ^= !(c ^ 45);
}
for (; isdigit(c); c = getchar())
{
x = (x << 1) + (x << 3) + (c ^ 48);
}
if (f)
{
x = -x;
}
return x;
}
inline void write(int x)
{
if (x < 0)
{
putchar('-');
x = -x;
}
if (x > 9)
{
write(x / 10);
}
putchar(x % 10 + '0');
}
int n, m;
int c[700005];
int d[700005];
int ans;
inline int lowbit(int x)
{
return (x & -x);
}
inline void add(int x)
{
for (; x <= n; x += lowbit(x))
c[x]++;
}
inline int sum(int x)
{
int s = 0;
for (; x; x -= lowbit(x))
s += c[x];
return s;
}
int get()
{
int res = 0;
for (int i = n; i >= 1; i--)
add(d[i]), res += sum(d[i] - 1);
return res & 1;
}
signed main()
{
cin >> n >> m;
for (int i = 1; i <= n; ++i)
cin >> d[i];
ans = get();
for (int i = 1, z, a, b, c; i <= m; ++i)
{
z = read();
a = read();
b = read();
if (a > b)
swap(a, b);
if (z == 1)
{
if (a != b)
ans ^= 1;
}
else if (z == 2)
{
if ((len >> 1) & 1)
ans ^= 1;
}
else if (z == 3)
{
c = read();
if ((c * (len - 1)) & 1)
ans ^= 1;
}
else
{
c = read();
if ((c * (len - 1)) & 1)
ans ^= 1;
}
write(ans);
putchar('\n');
}
}
本文来自博客园,作者:蒟蒻orz,转载请注明原文链接:https://www.cnblogs.com/orzz/p/18122178