方法一:暴力
class Solution {
public int[] smallerNumbersThanCurrent(int[] nums) {
int n = nums.length;
int[] ret = new int[n];
for (int i = 0; i < n; i++) {
int cnt = 0;
for (int j = 0; j < n; j++) {
if (nums[j] < nums[i]) {
cnt++;
}
}
ret[i] = cnt;
}
return ret;
}
}
方法二:快速排序
我们也可以将数组排序,并记录每一个数在原数组中的位置。对于排序后的数组中的每一个数,我们找出其左侧第一个小于它的数,这样就能够知道数组中小于该数的数量。
class Solution {
public int[] smallerNumbersThanCurrent(int[] nums) {
int n = nums.length;
int[][] data = new int[n][2];
for (int i = 0; i < n; i++) {
data[i][0] = nums[i];
data[i][1] = i;
}
Arrays.sort(data, new Comparator<int[]>() {
public int compare(int[] data1, int[] data2) {
return data1[0] - data2[0];
}
});
int[] ret = new int[n];
int prev = -1;
for (int i = 0; i < n; i++) {
if (prev == -1 || data[i][0] != data[i - 1][0]) {
prev = i;
}
ret[data[i][1]] = prev;
}
return ret;
}
}
方法三:计数排序
注意到数组元素的值域为 [0,100],所以可以考虑建立一个频次数组 cnt ,cnt[i] 表示数字 i 出现的次数。那么对于数字 i 而言,小于它的数目就为 cnt[0…i-1] 的总和。
class Solution {
public int[] smallerNumbersThanCurrent(int[] nums) {
int[] cnt = new int[101];
int n = nums.length;
for (int i = 0; i < n; i++) {
cnt[nums[i]]++;
}
for (int i = 1; i <= 100; i++) {
cnt[i] += cnt[i - 1];
}
int[] ret = new int[n];
for (int i = 0; i < n; i++) {
ret[i] = nums[i] == 0 ? 0 : cnt[nums[i] - 1];
}
return ret;
}
}