题解:士兵排列

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;
}
相关推荐
寒秋花开曾相惜8 小时前
(学习笔记)第四章 处理器体系结构
linux·网络·数据结构·笔记·学习
故事和你918 小时前
洛谷-数据结构1-1-线性表1
开发语言·数据结构·c++·算法·leetcode·动态规划·图论
脱氧核糖核酸__8 小时前
LeetCode热题100——53.最大子数组和(题解+答案+要点)
数据结构·c++·算法·leetcode
脱氧核糖核酸__9 小时前
LeetCode 热题100——42.接雨水(题目+题解+答案)
数据结构·c++·算法·leetcode
王老师青少年编程10 小时前
csp信奥赛C++高频考点专项训练之贪心算法 --【线性扫描贪心】:数列分段 Section I
c++·算法·编程·贪心·csp·信奥赛·线性扫描贪心
王老师青少年编程10 小时前
csp信奥赛C++高频考点专项训练之贪心算法 --【线性扫描贪心】:分糖果
c++·算法·贪心算法·csp·信奥赛·线性扫描贪心·分糖果
_日拱一卒10 小时前
LeetCode:2两数相加
算法·leetcode·职场和发展
py有趣10 小时前
力扣热门100题之零钱兑换
算法·leetcode
董董灿是个攻城狮10 小时前
Opus 4.7 来了,我并不建议你升级
算法
自我意识的多元宇宙10 小时前
二叉树遍历方式代码解读(2迭代)
数据结构