Q - Queue HDU - 5493

Problem Description NNN people numbered from 1 to NNN are waiting in a

bank for service. They all stand in a queue, but the queue never

moves. It is lunch time now, so they decide to go out and have lunch

first. When they get back, they don’t remember the exact order of the

queue. Fortunately, there are some clues that may help. Every person

has a unique height, and we denote the height of the iii-th person as

hih_ihi​. The iii-th person remembers that there were kik_iki​ people who

stand before him and are taller than him. Ideally, this is enough to

determine the original order of the queue uniquely. However, as they

were waiting for too long, some of them get dizzy and counted kik_iki​ in

a wrong direction. kik_iki​ could be either the number of taller people

before or after the iii-th person. Can you help them to determine the

original order of the queue?

Input The first line of input contains a number TTT indicating the

number of test cases (T≤1000T≤1000T≤1000). Each test case starts with a line

containing an integer NNN indicating the number of people in the queue

(1≤N≤1000001≤N≤1000001≤N≤100000). Each of the next NNN lines consists of two integers

hih_ihi​ and kik_iki​ as described above (1≤hi≤109,0≤ki≤N−11≤h_i≤10^9,0≤k_i≤N-11≤hi​≤109,0≤ki​≤N−1). Note that

the order of the given hih_ihi​ and kik_iki​ is randomly shuffled. The sum

of NNN over all test cases will not exceed 10610^6106

Output For each test case, output a single line consisting of “Case

#X: S”. XXX is the test case number starting from 1. SSS is people’s heights in the restored queue, separated by spaces. The solution may

not be unique, so you only need to output the smallest one in

lexicographical order. If it is impossible to restore the queue, you

should output “impossible” instead.

Sample Input
3
3
10 1
20 1
30 0
3
10 0
20 1
30 0
3
10 0
20 0
30 1 Sample Output
Case #1: 20 10 30
Case #2: 10 20 30
Case #3: impossible

思路

利用插空的思想, 当我们考虑 某个人 要被放到那个位置上的时候, 考到 可能是比他 高到k个人在他的前边,也可能是比他高到人在他的后边,这个是我们可 把题上给我们的序列按照从身高从小的到大来排序,这我正考虑要某个要放哪的时候是不会受 前边人已经安排好为位置的人影响,,这操作之后这个题目看起来简单些,,,接着由于字典序最小,,,由于每个人的身高是从小到大排序的,所以我们在 决定某个人要被放到 那个位置的时候(这个时候有多个位置符合题意) 我们应该尽量的把这个人放置的越靠前位置越好(在这个位置是符合题意的时候), 那么我们怎判断某个人因该放那个位置呢??我们可以假设 我们要放置的是 第 i 个人,在他的前边或后边 有k个人比他高, 可以确定已经有 i - 1 个人已经被的位置已经被确定了,我们先讨论第一个中情况:在 i 这个人的前边有 k 个人比他高, ⚠️之前放置的人,都比他矮,所以我们要预留 k + 1个空位置(因为 i 这个人也要占一个位置),具体找这个位置,我们可以通过 二分查找 来查找 前缀区间和为 k + 1 的位置 ; 接下来我们讨论第二种情况:在 区间的尾部留 k空位置,那么在这个空位置的前边剩余的空位置为 n - i - k + 1 , 这样在用二分 来查找 区间前缀和为 n - i - k + 1 的位置的下标,由于字典序最小我们应在两种情况的到的位置中取最小的情况,并在把这个位置 天上这个人的身高就行了。。。。。。。这样不断遍历循环讨论下去,就能得到答案了

代码

#include<stdio.h>
#include<string>
#include<string.h>
#include<cstdio>
#include<algorithm>
#include<iostream>
using namespace std;
typedef long long ll;
const int MAXN=200000+5; //最大元素个数
int n;//元素个数
int c[MAXN],ans[MAXN]; //c[i]==A[i]+A[i-1]+...+A[i-lowbit(i)+1] int lowbit(int i){return i&(-i);} //返回i的二进制最右边1的值 int sum(int x){ //返回A[1]+...A[i]的和
int sum = 0;
while(x){
sum += c[x];
x -= lowbit(x);
}
return sum;
} void add(int x, int val){ //令A[i] += val
while(x <= n){
c[x] += val;
x += lowbit(x);
}
} int find_(int x){ //二分查找
int l=1,r=n,mid;
while(l<r){
mid=(l+r)>>1;
int num=sum(mid);
if(num<x)
l=mid+1;
else
r=mid;
}
return l;
} int Sch(int num) //二分查找空位数为num 的位置
{
int l = 1, r = n, ans;
while(l <= r)
{
int mid = (l + r) >> 1;
if(Sum(mid) >= num)
r = mid - 1, ans = mid;
else
l = mid + 1;
}
return ans;
} struct node1{
ll h,v;
}a[MAXN]; ll b[MAXN]; bool cmp(node1 x,node1 y){
return x.h<y.h;
} int main(){
int t;
t--;
cin>>t;
int kase=0;
while(t--){
kase++;
memset(c,0,sizeof(c));
memset(ans,0,sizeof(ans));
scanf("%lld",&n);
for(int i=1;i<=n;i++){
add(i,1);
scanf("%lld%lld",&a[i].h,&a[i].v);
} sort(a+1,a+n+1,cmp);
int flag=1;
for(int i=1;i<=n;i++){
if(n-i-a[i].v<0) {flag=0;break;}
int p=min(a[i].v,n-i-a[i].v)+1; //前面有多少个空位,包括本身,所以+1
int pos=find_(p); add(pos,-1);
ans[pos]=a[i].h;
} printf("Case #%d:", kase);
if (flag) {
for (int i = 1; i <= n; i++) {
printf(" %d", ans[i]);
}printf("\n");
}
else printf(" impossible\n");
}
return 0;
}
  • 总结:1. 对待这种插空位置,我们应该按照一定的顺序,比如这一题 按身高从低到高 去一一的对待去讨论, 这样避免一些了复杂的东西。
  • 2.这里的 “字典序最小” 与 “预留空位的操作” ,是奇妙的搭配
  • 3.这一题的 树状数组 用来统计 空位置 的数量, 而一个又一个讨论找符合题意的下标位置 是很浪费时间的,所以加上了 一个快速 “二分” , 快速查找就非常完美了

Q - Queue HDU - 5493(树状树组维护区间前缀和 + 二分找预留空位)的更多相关文章

  1. HDU 5877 dfs+ 线段树(或+树状树组)

    1.HDU 5877  Weak Pair 2.总结:有多种做法,这里写了dfs+线段树(或+树状树组),还可用主席树或平衡树,但还不会这两个 3.思路:利用dfs遍历子节点,同时对于每个子节点au, ...

  2. G - Queue HDU - 5493 线段树+二分

    G - Queue HDU - 5493 题目大意:给你n个人的身高和这个人前面或者后面有多少个比他高的人,让你还原这个序列,按字典序输出. 题解: 首先按高度排序. 设每个人在其前面有k个人,设比这 ...

  3. 牛客练习赛38 D 题 出题人的手环 (离散化+树状数组求逆序对+前缀和)

    链接:https://ac.nowcoder.com/acm/contest/358/D来源:牛客网 出题人的手环 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 524288K,其他 ...

  4. 关于线段树or 树状树状 在二维平面搞事情!Orz

    第一式:https://ac.nowcoder.com/acm/contest/143/I 题意: 有 n 个点,一个点集 S 是好的,当且仅当对于他的每个子集 T,存在一个右边无限长的矩形,使得这个 ...

  5. 「BZOJ1537」Aut – The Bus(变形Dp+线段树/树状数组 最优值维护)

    网格图给予我的第一反应就是一个状态 f[i][j] 表示走到第 (i,j) 这个位置的最大价值. 由于只能往下或往右走转移就变得显然了: f[i][j]=max{f[i-1][j], f[i][j-1 ...

  6. 洛谷 P3368 【模板】树状数组 2(区间加,单点查询)

    题目链接 https://www.luogu.org/problemnew/show/P3368 树状数组 最基础的用法:https://www.cnblogs.com/yinyuqin/p/1096 ...

  7. A Simple Problem with Integers 多树状数组分割,区间修改,单点求职。 hdu 4267

    A Simple Problem with Integers Time Limit: 5000/1500 MS (Java/Others)    Memory Limit: 32768/32768 K ...

  8. HDU 5465——Clarke and puzzle——————【树状数组BIT维护前缀和+Nim博弈】

    Clarke and puzzle Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others ...

  9. 主席树套树状数组——带修区间第k大zoj2112

    主席树带修第k大 https://www.cnblogs.com/Empress/p/4659824.html 讲的非常好的博客 首先按静态第k大建立起一组权值线段树(主席树) 然后现在要将第i个值从 ...

随机推荐

  1. 最适合初学者的一篇 Ribbon 教程

    什么是 Ribbon Ribbon 是一个基于 HTTP 和 TCP 的 客服端负载均衡工具,它是基于 Netflix Ribbon 实现的. 它不像 Spring Cloud 服务注册中心.配置中心 ...

  2. 点云3d检测模型pointpillar

    PointPillars 一个来自工业界的模型.https://arxiv.org/abs/1812.05784 3D目标检测通常做法 3d卷积 投影到前平面 在bird-view上操作 处理思路依然 ...

  3. Python入门的三大问题和三大谎言

    Python广告,铺天盖地,小白们雾里看花,Python无限美好.作为会20几种语言(BASIC Foxbase/pro VB VC C C++ c# js typescript HTML Ardui ...

  4. 0312 java接口测试三棱军刺rest-assured

    背景 java程序员一般写的是后端服务是JavaWeb类型的项目,主要包括Http接口和dubbo接口,Http接口一般采用的rest风格,那么如何快速的对rest接口在第三方的测试框架上进行测试呢? ...

  5. BFC块级格式上下文介绍

    块级格式上下文(Block formatting context) 什么是BFC? 块格式化上下文(block formatting context) 是页面 CSS视觉渲染的一部分.它是用于决定块盒 ...

  6. pytest、tox、Jenkins实现python接口自动化持续集成

    pytest介绍 pytest是一款强大的python测试工具,可以胜任各种级别的软件测试工作,可以自动查找测试用并执行,并且有丰富的基础库,可以大幅度提高用户编写测试用例的效率,具备可扩展性,用户自 ...

  7. ggplot2(9) 数据操作

    9.1 plyr包简介 ddply {plyr}: Split data frame, apply function, and return results in a data frame. ddpl ...

  8. 题解 UVA12186 【工人的请愿书 Another Crisis】

    俺太难了 记录一下我调了一个小时的错误 多测不清空 多测清空只清空了\(vector\) 多测全清空了,但是忘了清空\(vector[0]\) \(priority\)_ \(queue\)把\(gr ...

  9. CentOS7使用tar.gz方法安装php7.2.8

    软件:VMware 系统版本:CentOS7 1.检查系统里是否有安装的PHP包,如果有的话先删除 [root@localhost ~]# yum list installed | grep php ...

  10. Css盒模型属性详解(margin和padding)

    Css盒模型属性详解(margin和padding) 大家好,我是逆战班的一名学员,今天我来给大家分享一下关于盒模型的知识! 关于盒模型的属性详解及用法 盒模型基本属性有两个:padding和marg ...