Mato的文件管理 bzoj-3289

题目大意:给定一个n个数的序列。m次询问:一段区间中的逆序对个数。

注释:$1\le n\,mle 5\cdot 10^4$。


想法

开始想这个题的大佬们,给您点儿提示吧:$O(nlogn\sqrt(n))$可过哦!

所以这个题就是莫队的裸题了。

我们的莫队上的区间在动的时候随时更新树状数组上的信息即可。、

然后碰见了一整块区间,我们就直接求逆序对即可,

最后,附上丑陋的代码... ...

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#define N 50010
using namespace std;
struct Query {int l,r,id;}q[N]; int L[N],R[N],maxval,tree1[N<<1],tree2[N<<1],blg[N],a[N],ans[N];
inline bool cmp_dispose(const Query &x,const Query &y) {return blg[x.l]==blg[y.l]?x.r<y.r:blg[x.l]<blg[y.l];}
inline char nc() {static char *p1,*p2,buf[100000]; return (p1==p2)&&(p2=(p1=buf)+fread(buf,1,100000,stdin),p1==p2)?EOF:*p1++;}
int rd() {int x=0; char c=nc(); while(!isdigit(c)) c=nc(); while(isdigit(c)) x=(x<<3)+(x<<1)+(c^48),c=nc(); return x;}
inline int lowbit(int x) {return x&(-x);}
void update1(int x,int val) {for(int i=x;i>=1;i-=lowbit(i)) tree1[i]+=val;}
void update2(int x,int val) {for(int i=x;i<=maxval;i+=lowbit(i)) tree2[i]+=val;}
inline void update(int x,int val) {update1(x,val); update2(x,val);}
int query1(int x) {int ans=0; for(int i=x;i<=maxval;i+=lowbit(i)) ans+=tree1[i]; return ans;}
int query2(int x) {int ans=0; for(int i=x;i>=1;i-=lowbit(i)) ans+=tree2[i]; return ans;}
int main()
{
int n=rd(); int t=sqrt(n); int blck=n/t;
for(int i=1;i<=blck;i++)
{
int bfr=(i-1)*t;
L[i]=bfr+1; R[i]=bfr+t;
for(int j=1;j<=t;j++)
{
blg[bfr+j]=i;
a[bfr+j]=rd();
maxval=max(maxval,a[bfr+j]);
}
}
if(blck*t<n)
{
L[blck+1]=blck*t; R[blck+1]=n; blck++;
for(int j=blck*t+1;j<=n;j++)
{
blg[j]=blck; a[j]=rd(); maxval=max(maxval,a[j]);
}
}
int m=rd();
for(int i=1;i<=m;i++) q[i].l=rd(),q[i].r=rd(),q[i].id=i;
sort(q+1,q+m+1,cmp_dispose);
int point_l=q[1].l,point_r=q[1].r; for(int i=point_l;i<=point_r;i++)
{
ans[q[1].id]+=query1(a[i]+1); update(a[i],1);
}
for(int i=2;i<=m;i++)
{
int idx=q[i].id;
while(point_l<q[i].l)
{
update(a[point_l],-1);
ans[idx]-=query2(a[point_l]-1);
point_l++;
}
while(point_l>q[i].l)
{
point_l--;
update(a[point_l],1);
ans[idx]+=query2(a[point_l]-1);
}
while(point_r<q[i].r)
{
point_r++;
update(a[point_r],1);
ans[idx]+=query1(a[point_r]+1);
}
while(point_r>q[i].r)
{
update(a[point_r],-1);
ans[idx]-=query1(a[point_r]+1);
point_r--;
}
}
for(int i=1;i<=m;i++) printf("%d\n",ans[i]);
return 0;
}

小结:莫队真的强..

[bzoj3289]Mato的文件管理_莫队_树状数组的更多相关文章

  1. BZOJ3289 Mato的文件管理(莫队算法+树状数组)

    题目是区间逆序数查询. 莫队算法..左或右区间向左或右延伸时加或减这个区间小于或大于新数的数的个数,这个个数用树状数组来统计,我用线段树超时了.询问个数和数字个数都记为n,数字范围不确定所以离散化,这 ...

  2. 【bzoj3289】Mato的文件管理 离散化+莫队算法+树状数组

    原文地址:http://www.cnblogs.com/GXZlegend/p/6805224.html 题目描述 Mato同学从各路神犇以各种方式(你们懂的)收集了许多资料,这些资料一共有n份,每份 ...

  3. 【BZOJ】3289: Mato的文件管理(莫队算法+树状数组)

    http://www.lydsy.com/JudgeOnline/problem.php?id=3289 很裸的莫队... 离线了区间然后分块排序后,询问时搞搞就行了. 本题中,如果知道$[l, r] ...

  4. BZOJ 3289:Mato的文件管理(莫队算法+树状数组)

    http://www.lydsy.com/JudgeOnline/problem.php?id=3289 题意:…… 思路:求交换次数即求逆序对数.确定了这个之后,先离散化数组.然后在后面插入元素的话 ...

  5. HDU6534 Chika and Friendly Pairs(莫队,树状数组)

    HDU6534 Chika and Friendly Pairs 莫队,树状数组的简单题 #include<bits/stdc++.h> using namespace std; cons ...

  6. 【BZOJ3289】Mato的文件管理 莫队算法+树状数组

    [BZOJ3289]Mato的文件管理 Description Mato同学从各路神犇以各种方式(你们懂的)收集了许多资料,这些资料一共有n份,每份有一个大小和一个编号.为了防止他人偷拷,这些资料都是 ...

  7. BZOJ3289 Mato的文件管理 【莫队 + 树状数组】

    3289: Mato的文件管理 Time Limit: 40 Sec  Memory Limit: 128 MB Submit: 3964  Solved: 1613 [Submit][Status] ...

  8. BZOJ 3289: Mato的文件管理[莫队算法 树状数组]

    3289: Mato的文件管理 Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 2399  Solved: 988[Submit][Status][Di ...

  9. BZOJ3289【莫队算法+树状数组+离散化】

    思路: 区间逆序数即是交换次数. 逆序数,可以用树状数组吧. 怎么处理区间变换的时候求逆序数啊.. 这里分成左边的增/删,右边的增/删 因为是按时序插入, 所以左边增,增一个数,计算:ans+=sun ...

  10. HDU 6278 - Just h-index - [莫队算法+树状数组+二分][2018JSCPC江苏省赛C题]

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6278 Time Limit: 6000/3000 MS (Java/Others) Memory Li ...

随机推荐

  1. MongoDB的用户权限管理

    1.创建用户并授权语法:db.createUser({user:"UserName",pwd:"Password",roles:[{role:"Rol ...

  2. [转]C++常用字符串分割方法实例汇总

    本文实例汇总了C++常用字符串分割方法,分享给大家供大家参考.具体分析如下: 我们在编程的时候经常会碰到字符串分割的问题,这里总结下,也方便我们以后查询使用. 一.用strtok函数进行字符串分割 原 ...

  3. Windows8.1查看已连接无线WIFI密码

    Windows8.1操作系统下查看已连接无线wifi密码操作步骤如下: 1.右键任务栏中的无线图标,在弹出的菜单中选择"打开网络和共享中心": 2.在网络和共享中心界面中点击&qu ...

  4. 转 MySQL实验(三) 过程式数据库对象的使用

    转 http://blog.csdn.net/anne999/article/details/70432558

  5. scala学习笔记2:面向对象编程部分基础

    以下主要记录的是看完scala in programming这本书Functional Objects(第六章)后的要点总结. 1,程序中可变对象(var)和不可变对象(val)使用的权衡问题 不可变 ...

  6. Java 8 (7) 重构、测试和调试

    为改善可读性和灵活性重构代码 看到这里我们已经可以使用lambda和stream API来使代码更简洁,用在新项目上.但大多数并不是全新的项目,而是对现有代码的重构,让它变的更简洁可读,更灵活. 改善 ...

  7. WCF学习笔记(2)-WCF的通讯过程

    一.WCF中的ABC 场景:公司让你送一份合同文件,送文件的过程你可以选择的交通方式有打的,地铁或公交. 到了对方公司后,你要找到某负责人,并且要一份收到合同文件的回执和相应文件 要完成这项工作任务主 ...

  8. leetcode523 Continuous Subarray Sum

    思路: 令sum[p]表示p位置的前缀和.如果sum[i] % k == sum[j] % k (j - i > 1),则存在子段(i, j]的和能够整除k. 实现: class Solutio ...

  9. 2) 十分钟学会android--建立第一个APP,执行Android程序

    通过上一节课创建了一个Android的Hello World项目,项目默认包含一系列源文件,它让我们可以立即运行应用程序. 如何运行Android应用取决于两件事情:是否有一个Android设备和是否 ...

  10. 处理sql锁死问题

    --SQL Server死锁的查询方法:   exec master.dbo.p_lockinfo 0,0; ---显示死锁的进程,不显示正常的进程   exec master.dbo.p_locki ...