17.3.18 数据结构 校内赛(rotinv)(rise)
来源:互联网 发布:历年三大产业 数据 编辑:程序博客网 时间:2024/06/08 19:29
这套题是知名的idy002出的,我们来看看题:
1.rotinv
【解题报告】
这道题是这样一个思路:
首先我们需要将[1,n]的数组开一个镜像,在[n+1,2n]的空间存储相同的内容。我们先算出[1,n]之内的逆序对个数,之后将这个区间向右移动,算出这个新的区间的逆序对个数(这样做就可以穷尽所有的组合情况),之后输出就可以了。
那么我们用什么办法算出一个区间的逆序对个数呢?我们从左到右枚举每一个数,找出比这个数大的数累加个数就可以了。
那么怎么维护这样一个区间的答案呢?我们算出了[i,j]过后,右移一位即是[i+1,j+1],我们只需要把[i+1,j]中比a[j+1]大的数,再减去[i+1,j]中比a[i]小的个数(实质上就是把比原来的右端点小的减去,把比现在的右端点大的加上)。再把这样一个值同[i,j]的值加起来。
此外再用树状数组优化。
来看看代码:
#include<cstdio>#include<iostream>using namespace std;const int N = 1000000 + 10;template<class T>inline void readin(T &resez)//读入优化 { static char ch; while((ch=getchar())<'0'||ch>'9');resez=ch-48; while((ch=getchar())>='0'&&ch<='9')resez=resez*10+ch-48;}int n;int a[N+N];int bit[N*24];long long ans,cnt;void build(int pos,int delta)//在树状数组中添加 { for(register int i=pos;i<=n;i+=i&-i) bit[i]+=delta;}int query(int right)//树状数组中求和 { int rt=0; for(register int i=right;i;i-=i&-i) rt+=bit[i]; return rt;}int main() { freopen("rotinv.in","r",stdin); freopen("rotinv.out","w",stdout); readin(n); for(register int i=1;i<=n;i++) { readin(a[i]); a[n+i]=a[i]; } for(register int i=1;i<=n;i++) { cnt+=(i-1)-query(a[i]); build(a[i],+1); }//这时cnt中存储的是[1,n]中的逆序对个数 for(register int i=n+1;i<=n+n;i++)//这里就将a[1,n]的区间向右移,减去比a[i-n](原右端点)小的,加上比a[i](现右端点)大的 { build(a[i-n],-1); cnt+=n-1-query(a[i]);//比a[i]大的数 cnt-=query(a[i-n]-1);//比a[i-n]小的数 build(a[i],+1); ans+=cnt; } printf("%d",ans); return 0;}
2.rise
【解题报告】
这道题有两种解法,一种是用线段树(此处略),另一种是通过搞一个类似于“链表”的结构,将某一柱子向右离他最近的那个柱子标记好。最后查询时按照我构造的这个路径即可。
#include<cstdio>#include<iostream>#define N 100010using namespace std;template<class T>inline void read(T &res)//读入优化 { static char ch; while((ch=getchar())<'0'||ch>'9'); res=ch-48; while((ch=getchar())>='0'&&ch<='9') res =ch-48+res*10;}int later[N];int a[N];int main(){ freopen("rise.in","r",stdin); freopen("rise.out","w",stdout); int n,q; read(n),read(q); for(register int i=1;i<=n;i++) read(a[i]); for(register int i=1;i<=n;i++) for(register int j=i+1;j<=n;j++) if (a[j]>a[i])//(预处理)暴力枚举找到数组中比a[i]大的最近的那个数 { later[i]=j; break; } while(q--) { int l,r; read(l),read(r); int p=l; int tot=0; while(p<=r&&p!=0)//这里从l开始,依据预处理的路径来计数 { tot++; p=later[p]; } printf("%d\n",tot); }}
以上
2017.3.23
0 0
- 17.3.18 数据结构 校内赛(rotinv)(rise)
- 校内赛数据结构(rise,rotinv,seqmod)
- 17.4.4 数据结构 校内赛(setmod)(area)(intkth)
- 校内新生赛(网络赛)反思
- (未完成)校内赛得失谈
- 校内一(1)
- 校内选拔(六)
- Android 闹钟(One Rise)
- 【校内】快速幂(数论)
- [校内测试]祖先(DP)
- HDU2010省赛集训队选拔赛(校内赛) H
- HDU2010省赛集训队选拔赛(校内赛) G
- HFOI2017.07.09校内赛(普及组)题解
- HFOI2017.07.10校内赛(普及组)题解
- HFOI2017.07.11校内赛(普及组)题解
- HFOI2017.07.11校内赛(普及组)题解
- HFOI2017.7.13校内赛(普及组)题解
- HFOI2017.7.14校内赛(普及组)题解
- 最大公约数和最小公倍数
- CSDN学霸课表——网络工程师(软考中级)
- spring事务管理方式
- 《ACM程序设计》N题
- echarts饼状图的创建
- 17.3.18 数据结构 校内赛(rotinv)(rise)
- web前端-在迷惘中的探索HTML5(三)文件操作FileReader
- 封装DLL到EXE
- handler机制 源码分析 梳理
- Why bootstrap/How to use bootstrap
- C语言学习笔记7
- 沉没的王国---揭秘滇东自杞国(2)
- 读取流域平均降雨累计值
- Sip 响应状态码 对照 详解 【转】