bzoj 3333: 排队计划 解决问题的方法

【原标题】

3333: 排队计划

Time Limit: 20 Sec  Memory Limit: 128 MB

Submit: 161  Solved: 71

[Submit][Status]

Description

Input

Output

Sample Input

6 2

160 163 164 161 167 160

2

3

Sample Output

6

3

1

HINT

Source

wyx528命题

【分析】简述一下题目。N个数排成一列。每次指定一个位置P,然后把P~N中全部身高小于等于P的人都拎出来,排一遍序后再放进去。每次操作须要求一遍逆序对。

首先非常easy想到:每次选定一个位置P后,降低的逆序对数量就是P~N中满足要求的人原来构成的逆序对——并且当操作完毕之后。这些人将永远不会产生逆序对了。

考虑到这种性质,我们能够维护一个数的F[i],表示从I这个位置開始的逆序对数量。(我们并不关心某个位置某一个时刻的值是多少)然后我们用一个线段树维护i~N中的最小值。每次操作寻找P~N的最小值。假设是满足要求的且位置是X,那么我们就把f[x]置为0,然后能够把x位置的数置为无穷大。之后继续寻找,直到找不到为止。由于每一个数仅仅会最多被更新一次,所以效率均摊是NlogN的。

【代码】

#include<cstdio>
#include<algorithm>
#define N 500005
#define INF 1000000005
#define Lo(x) (x&-x)
using namespace std;
typedef long long LL;LL ans=0;
struct Tree{int l,r,min,wh;}a[N*3];
int tree[N];//pos P in array to the tree
int pos[N*4];//pos P int tree to the array
int f[N],data[N],c[N],n,cnt,x,i,opt,P,ord,size,now,L,R;
struct array{int x,id;}uni[N];
inline int cmp(const array &a,const array &b){return a.x<b.x;}
inline int ask(int x){int s=0;for (;x;x-=Lo(x)) s+=c[x];return s;}
inline void add(int x){for (;x<=n;x+=Lo(x)) c[x]++;}
void build(int k,int l,int r)
{
  a[k].l=l;a[k].r=r;
  if (l==r) {a[k].min=data[l];pos[k]=l;tree[l]=k;a[k].wh=k;return;}
  int mid=(l+r)>>1;
  if (l<=mid) build(k<<1,l,mid);
  if (r>mid) build(k<<1|1,mid+1,r);
  if (a[k<<1].min<a[k<<1|1].min) a[k].min=a[k<<1].min,a[k].wh=a[k<<1].wh;
  else a[k].min=a[k<<1|1].min,a[k].wh=a[k<<1|1].wh;
}
int query(int k)
{
  if (L<=a[k].l&&a[k].r<=R) return a[k].wh;
  int mid=(a[k].l+a[k].r)>>1,resl=0,resr=0;
  if (L<=mid) resl=query(k<<1);
  if (R>mid) resr=query(k<<1|1);
  if (!resl) return resr;if (!resr) return resl;
  return (a[resl].min<a[resr].min)?

resl:resr;
}
void update(int k)
{
  if (a[k].l==a[k].r) {a[k].min=INF;a[k].wh=k;return;}
  int mid=(a[k].l+a[k].r)>>1;
  if (ord<=mid) update(k<<1);else update(k<<1|1);
  if (a[k<<1].min<a[k<<1|1].min) a[k].min=a[k<<1].min,a[k].wh=a[k<<1].wh;
  else a[k].min=a[k<<1|1].min,a[k].wh=a[k<<1|1].wh;
 }
int main()
{
  read(n);read(opt);//读入优化略去
  for (i=1;i<=n;i++) read(x),uni[i]=(array){x,i};
  sort(uni+1,uni+n+1,cmp);
  for (i=1;i<=n;i++)
    data[uni[i].id]=(uni[i].x==uni[i-1].x)?cnt:++cnt;
  for (i=n;i;i--)
    f[i]=ask(data[i]-1),add(data[i]),ans+=(LL)f[i];
  build(1,1,n);printf("%lld\n",ans);
  while (opt--)
  {
    read(P);now=a[tree[P]].min;
    while (now<INF)
    {
      L=P;R=n;ord=query(1);size=a[ord].min;
      if (size>now) break;ord=pos[ord];
      ans-=(LL)f[ord];f[ord]=0;update(1);
    }
    printf("%lld\n",ans);
  }
  return 0;
} 

版权声明:本文博主原创文章,博客,未经同意不得转载。

时间: 2024-10-27 12:52:34

bzoj 3333: 排队计划 解决问题的方法的相关文章

bzoj 3333: 排队计划 题解

[原题] 3333: 排队计划 Time Limit: 20 Sec  Memory Limit: 128 MB Submit: 161  Solved: 71 [Submit][Status] Description Input Output Sample Input 6 2 160 163 164 161 167 160 2 3 Sample Output 6 3 1 HINT Source wyx528命题 [分析]简述一下题目.N个数排成一列,每次指定一个位置P,然后把P~N中所有身高小

BZOJ 3333: 排队计划 树状数组+线段树

题目大意:给出一个序列,求出这个序列的逆序对数量.定义一种操作,将一个数和他后面比他小的数字拿出来排序, 然后再放回去,之后输出逆序对数. 思路:思路题.手动模拟一下,会发现,逆序对变化的只是排序的那些点 .所以我们只要处理那些点就行了.先求一次逆序对,然后每次在拿出的数后面找到一个最小的数字,把它的权值改成INF,统计答案. CODE: #include <cstdio> #include <cstring> #include <iostream> #include

BZOJ 2595 游览计划(插头DP)

题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=2595 题意:给出一个数字矩阵.求一个连通块包含所有的数字0且连通块内所有数字之和最小. 思路:对于每个格子,是0则必须要选.那么对于不选的格子(i,j)在什么时候可以不选呢?必须同时满足以下两个条件: (1)(i,j)不是0: (2)(i-1,j)不选或者(i-1,j)选了但是轮廓线上还有别的地方与(i-1,j)是一个连通块. int Pre[105][N],op[105][N]; s

《成为技术领导者-掌握全面解决问题的方法》读后感

<成为技术领导者-掌握全面解决问题的方法>的作者是美国作家Gerald M.Weinberg,译者余晟. 我花了大约一周的时间将本书阅读完成的,购买本书的也是一个很巧合的,因为看到书名包含了技术(联想到了编程技术,因为作者从事过软件开发),且浏览了目录像是有技巧性的知识包含其中,于是就购买了,结果阅读的过程当中,我才发现本书跟编程技术.技巧没有多少关系,而副标题才是准确表达本书的核心,全书分为定义.创新.激励.组织.转变五个部分,详细入微的讨论了一个技术领导要解决问题的各种方法,让你看到,技术

【bzoj 3333】排队计划(线段树)

n个数,求一次逆序对.接着有m次修改操作,把每次输入的位置p的数之后<=它的数取出来,从小到大排序后再放回空位里,求逆序对.(N,M<=500,000 , Ai<=10^9)思路:1.往后修改就存后缀,而不是一般的前缀.存数 i 之后<=它的数的个数为s[i],用于后续求逆序对.2.修改时选出的数排序后,它们的 s[] 都清零了,也可以“删掉”它们了——更改其值为INF.实现:1.用树状数组(或线段树)求出初始的逆序对数 sum.2.每次操作用线段树在p到n的区间内找到所有<

边坡优化主题5——bzoj 1096 [ZJOI2007]仓库建设 解决问题的方法

[原标题] 1096: [ZJOI2007]仓库建设 Time Limit: 10 Sec  Memory Limit: 162 MB Submit: 1998  Solved: 816 [id=1096" style="color:blue; text-decoration:none">Submit][Status] Description L公司有N个工厂,由高究竟分布在一座山上. 如图所看到的,工厂1在山顶.工厂N在山脚. 因为这座山处于高原内陆地区(干燥少雨),

1月31日 解决问题的方法( 麦肯锡七步成诗法 )

第一步:问题描述  1 明确企业要解决的基本问题  2 具体的.有内容的描述问题  3 清楚列示问题涉及的各方面信息 第二步:问题的分解 1 为何要进行分解  a 分解是提出假设的基础          提出假设          搜集资料          分析论证假设          完成咨询报告  b 理清思路         分解区分         设置优先顺序 2 问题分解的原则  a 内容是不是全面充分?  b 分解后的要素是不是相互独立? 3 问题分解的方法  a 不断提出假设

Hostspace组织培训活动——解决问题,方法先行

在日常工作中,大部分工作内容都是在"解决问题",解决问题是一个不断摆事实,找原因,提方案的过程.简单的问题依靠直线思维和经验判断就可解决,但碰上复杂问题时,常常由于主观认知先入为主或思考问题时缺少合理假设等原因导致头脑发热,被一些问题表面现象所迷惑,抓不住主要原因和问题本质.导致复杂问题简单化,想要考虑全面,却往往适得其反.总结起来,是我们在工作中缺少一套科学的分析与解决问题体系.完整的解决问题框架. 为了对公司成员解决问题的能力进行系统化的训练和提升,更好地履行各自的岗位职责,提高产

[Oracle] 获取运行计划的各方法总结

总的结论: 一.获取运行计划的6种方法(具体步骤已经在每一个样例的开头凝视部分说明了): 1. explain plan for获取: 2. set autotrace on . 3. statistics_level=all; 4. 通过dbms_xplan.display_cursor输入sql_id參数直接获取 5. 10046 trace跟踪 6. awrsqrpt.sql 二.适用场合分析 1.假设某SQL运行很长时间才会出结果.甚至慢到返回不了结果,这时候看运行计划就仅仅能用方法1,