题解:士兵排列

Description

军士长Johnny手下一共有n个士兵,每个士兵按照身高从矮到高编号为1到n。可惜这群士兵太笨了,第一次排好后现在又忘记怎么排了。站在Johnny面前的士兵虽然排成了一列,但是顺序就不一定是从1到n了。

Johnny想了如下的一个办法让他们排列整齐: 从最前面的士兵开始,每个士兵一直往前排直到碰到一个编号比自己小的士兵(身高小于自己)为止,然后该士兵就站在这个位置。不一会儿,队伍调整好了。 Johnny觉得很满意,不过为了提高效率,他要士兵们记住,这次调整的过程中,每个人各自往前走了几个人的位置。 现在给你这样一个任务:告诉你在排队过程中每个人往前走了几个人的位置,请你告诉我最初这个队伍的顺序是怎样的。

Input

输入第一行包含一个正整数n(1 ≤ n ≤ 200000),代表士兵个数。
第二行一共n个整数,第i个数代表原来队伍中排在第i个位置的人在调整过程中往前走了几个人的位置。

Output

输出包含一行,共n个整数,从左往右第i个数代表最初在队伍中第i个人的编号。相邻两个整数用一个空格隔开。

复制代码
5 
0 
复制代码
1 2 0 1 
Sample Input
复制代码
3 2 1 5 4
Sample Output
HINT

【数据限制】

有10%的数据,n不超过10。

有30%的数据,n不超过1000。

时间限制为1秒,空间限制为256MB。

思路:

模拟,树状数组优化

代码:

cpp 复制代码
#include <bits/stdc++.h>
#define int long long
using namespace std;

class BIT {
private:
    vector<int> tree;
    int size;
public:
    BIT(int n) : size(n), tree(n + 1, 0) {}
    void update(int idx, int delta) {
        while (idx <= size) {
            tree[idx] += delta;
            idx += idx & -idx;
        }
    }
    int query(int idx) {
        int sum = 0;
        while (idx > 0) {
            sum += tree[idx];
            idx -= idx & -idx;
        }
        return sum;
    }
    int findKth(int k) {
        int low = 1, high = size, ans = 0;
        while (low <= high) {
            int mid = (low + high) / 2;
            int s = query(mid);
            if (s >= k) ans = mid,high = mid - 1;
			else low = mid + 1;
        }
        return ans;
    }
};
signed main() {
    int n;
    cin >> n;
    vector<int> a(n);
    for (int i = 0; i < n; i ++) {
        cin >> a[i];
    }
    BIT bit(n);
    for (int i = 1; i <= n; i ++) bit.update(i, 1);
    vector<int> res(n + 1, 0);
    for (int i = n; i >= 1; i --) {
        int k = i - a[i - 1];
        int x = bit.findKth(k);
        res[i] = x;
        bit.update(x, -1);
    }
    for (int i = 1; i <= n; i ++) {
        if (i > 1) cout << ' ';
        cout << res[i];
    }
    return 0;
}
相关推荐
yasuniko2 分钟前
C++线程库
开发语言·c++
还有几根头发呀17 分钟前
深入理解C/C++内存管理:从基础到高级优化实践
c++
蒟蒻小袁1 小时前
力扣面试150题-- 翻转二叉树
算法·leetcode·面试
养一只Trapped_beast1 小时前
【LeetCode】删除排序数组中的重复项 II
算法·leetcode·职场和发展
矢鱼1 小时前
单调栈所有模版(2)
算法
轮到我狗叫了1 小时前
力扣智慧思想小题,目录力扣.跳跃游戏(思想很重要)力扣.跳跃游戏II(还是思想)力扣.分发糖果力扣151.反转字符串中的单词力扣.轮转数组
数据结构·算法·leetcode
zxctsclrjjjcph1 小时前
【递归、搜索和回溯】递归、搜索和回溯介绍及递归类算法例题
开发语言·c++·算法·力扣
朱剑君1 小时前
排序算法——堆排序
算法·排序算法
10000hours2 小时前
【SGL】Scatter-Gather List内存传输技术
linux·数据结构·网络协议·list·存储·sgl
某不知名網友2 小时前
linux_进程地址空间(虚拟地址空间)
java·linux·算法