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;
}