【题目链接】:http://codeforces.com/problemset/problem/785/E

【题意】



给你一个初始序列1..n顺序

然后每次让你交换任意两个位置上面的数字;

让你实时输出q个操作,每个操作过后整个序列逆序对的个数;

【题解】



分块法;

分成根号n个块.

每个块按照数字升序排;

然后再用一个a数组具体记录每个位置上的数字;

找逆序对的方式如下:

对于交换l,r

查找l+1..r-1当中比a[l]小的数字,比a[l]大的数字;

查找l+1..r-1当中比a[r]小…比..大的数字;

根据这4个参数来更新逆序对的个数;

这就要求查询l+1..r-1当中比x小的数的个数;

对于belong[l]块,->belong[i]指的是第i个位置上面的数字对应的是第几个块

在l..R[belong[l]]当中找比x小的数->暴力枚举就好

在L[belong[r]]..r当中找比x小的数->也是暴力枚举;

然后在belong[l]+1..belong[r]-1这些块中,用二分(lower_bound)查找比x小的数的个数(因为是有序的,所以可以这样做);

然后如果a[l]< a[r]则再多递增一个逆序对否则减少1个

然后交换这两个数;

->有序的vector中交换,然后原数组a[i]中也要交换;

vector中的交换也可以用lower_bound实现;



【完整代码】

#include <bits/stdc++.h>
using namespace std;
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
#define LL long long
#define rep1(i,a,b) for (int i = a;i <= b;i++)
#define rep2(i,a,b) for (int i = a;i >= b;i--)
#define mp make_pair
#define ps push_back
#define fi first
#define se second
#define rei(x) scanf("%d",&x)
#define rel(x) scanf("%lld",&x)
#define ref(x) scanf("%lf",&x) typedef pair<int, int> pii;
typedef pair<LL, LL> pll; const int dx[9] = { 0,1,-1,0,0,-1,-1,1,1 };
const int dy[9] = { 0,0,0,-1,1,-1,1,-1,1 };
const double pi = acos(-1.0);
const int K = 500+20;//500个块
const int N = 2e5 + 100; int n,q,belong[N],l[K],r[K],a[N];
LL ans = 0;
vector <int> v[K]; void init()
{
int len = sqrt(n);
if ((len*len) != n)
{
len++;
}
rep1(i, 1, n)
{
belong[i] = ((i - 1) / len) + 1;
v[belong[i]].ps(i);
a[i] = i;
}
rep1(i, 1, len)
{
l[i] = len*(i - 1) + 1;
r[i] = len*i;
}
if ((len*len) != n)
r[len] = n;
} LL smaller(int L, int R, LL x)
{
if (L > R) return 0;
LL cnt = 0;
if (belong[L] == belong[R])
{
rep1(i, L, R)
{
if (a[i] < x)
cnt++;
}
return cnt;
} rep1(i, L, r[belong[L]])
{
if (a[i] < x)
cnt++;
} rep1(i, belong[L] + 1, belong[R] - 1)
{
LL pos = lower_bound(v[i].begin(), v[i].end(), x) - v[i].begin()+1;
cnt += pos;
} rep1(i, l[belong[R]], R)
if (a[i] < x)
cnt++;
return cnt;
} void change(int l, int r)
{
int x = belong[l],y = belong[r];
v[x].erase(lower_bound(v[x].begin(), v[x].end(), a[l]));
v[x].insert(upper_bound(v[x].begin(), v[x].end(), a[r]),a[r]); v[y].erase(lower_bound(v[y].begin(), v[y].end(), a[r]));
v[y].insert(upper_bound(v[y].begin(), v[y].end(), a[l]), a[l]);
swap(a[l], a[r]);
} int main()
{
//freopen("F:\\rush.txt", "r", stdin);
rei(n), rei(q);
init();
rep1(i, 1, q)
{
int l, r;
rei(l), rei(r);
if (l > r) swap(l, r);
if (l == r)
{
printf("%lld\n", ans);
continue;
}
LL temp = smaller(l + 1, r - 1, a[l]);
LL temp1 = r - 1 - (l + 1) + 1 - temp;
//temp个数比a[l]小 temp1个数比a[l]大
//之后a[l]会跑到r位置
ans -= temp; ans += temp1;
temp = smaller(l + 1, r - 1, a[r]);
temp1 = r - 1 - (l + 1) + 1 - temp;
//temp个数比a[r]小 temp1个数比a[r]大
//之后a[r]会跑到l位置
ans += temp; ans -= temp1;
if (a[l] < a[r])
ans++;
else
ans--;
change(l, r);
printf("%lld\n", ans);
}
//printf("\n%.2lf sec \n", (double)clock() / CLOCKS_PER_SEC);
return 0;
}

【codeforces 785E】Anton and Permutation的更多相关文章

  1. 【25.00%】【codeforces 584E】Anton and Ira

    time limit per test1 second memory limit per test256 megabytes inputstandard input outputstandard ou ...

  2. 【27.91%】【codeforces 734E】Anton and Tree

    time limit per test3 seconds memory limit per test256 megabytes inputstandard input outputstandard o ...

  3. 【29.89%】【codeforces 734D】Anton and Chess

    time limit per test4 seconds memory limit per test256 megabytes inputstandard input outputstandard o ...

  4. 【13.77%】【codeforces 734C】Anton and Making Potions

    time limit per test4 seconds memory limit per test256 megabytes inputstandard input outputstandard o ...

  5. 【81.37%】【codeforces 734B】Anton and Digits

    time limit per test1 second memory limit per test256 megabytes inputstandard input outputstandard ou ...

  6. 【77.39%】【codeforces 734A】Anton and Danik

    time limit per test1 second memory limit per test256 megabytes inputstandard input outputstandard ou ...

  7. 【codeforces 508B】Anton and currency you all know

    [题目链接]:http://codeforces.com/contest/508/problem/B [题意] 给你一个奇数; 让你交换一次数字; 使得这个数字变成偶数; 要求偶数要最大; [题解] ...

  8. 【codeforces 785D】Anton and School - 2

    [题目链接]:http://codeforces.com/contest/785/problem/D [题意] 给你一个长度为n的括号序列; 让你删掉若干个括号之后,整个序列变成前x个括号为左括号,后 ...

  9. 【codeforces 734F】Anton and School

    [题目链接]:http://codeforces.com/problemset/problem/734/F [题意] 给你两个数组b和c; 然后让你找出一个非负数组a满足题中所给关系; [题解] 有个 ...

随机推荐

  1. JSP-Runoob:JSP 点击量统计

    ylbtech-JSP-Runoob:JSP 点击量统计 1.返回顶部 1. JSP 点击量统计 有时候我们需要知道某个页面被访问的次数,这时我们就需要在页面上添加页面统计器,页面访问的统计一般在用户 ...

  2. [Swift通天遁地]六、智能布局-(5)给视图添加Align(对齐)和Fill(填充的约束以及Label的约束

    ★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★➤微信公众号:山青咏芝(shanqingyongzhi)➤博客园地址:山青咏芝(https://www.cnblogs. ...

  3. Python基础类型(二) str 字符串

    字符串str ' ' 字符串+ 都是字符串的时候才能相加 a = 'alex' b = 'wusir' print(a+b) #字符串拼接 字符串* 字符串和数字相乘 a = 6 b = 'alex' ...

  4. C# 工厂单例

     public class BusinessFactory    {        private static BusinessFactory instance = null;        pri ...

  5. 拼接html 的事件转义

    attach += "<div style='line-height: 10px;float: left;margin-left: 10px;' id='attach_" + ...

  6. windows怎么进如debug调试

    主要说一下64位Win7使用debug程序的方法 首先你要下载一个DOSBOX程序 这个程序是一个dos模拟器 这个程序的制作目的是运行经典的DOS游戏 -.- 下载地址:http://www.dos ...

  7. 【洛谷4158/BZOJ1296】[SCOI2009]粉刷匠(动态规划)

    题目:洛谷4158 分析: 这题一看就是动态规划. 可以看出,如果每个木条粉刷的次数是固定的,那么这些木条是互不干扰的,因此对于每个木条可以通过dp来求出把T次中的j次分配给这个木条时可以获得的最大正 ...

  8. Hadoop Hive概念学习系列之hive里的桶(十一)

    不多说,直接上干货!  Hive还可以把表或分区,组织成桶.将表或分区组织成桶有以下几个目的: 第一个目的是为看取样更高效,因为在处理大规模的数据集时,在开发.测试阶段将所有的数据全部处理一遍可能不太 ...

  9. Java系列学习(十三)-字符串

    1.字符串基础 概念:字符串本质是打包字符数组的对象,是java.lang.String类的实例 2.字符串的构造方法 public String() public String(byte[] byt ...

  10. Java系列学习(十)-包与权限修饰符

    1.形式参数和返回值的问题 (1)形式参数: A:类名:需要该类的对象 B:抽象类名:需要改类的子类对象 C:接口名:需要该接口的实现对象 (2)返回值类型: A:类名:抽象类名:返回的是该类的对象 ...