排序后,只需考虑相邻元素之差(Python/Java/C++/C/Go/JS/Rust)
把 $\textit{arr}$ 排序后,最小绝对差只能来自相邻元素(不相邻的元素之差更大)。
遍历 $\textit{arr}$ 中的相邻元素 $(x,y)$,设绝对差为 $\textit{diff}=y-x$,当前最小绝对差为 $\textit{minDiff}$。
- 如果 $\textit{diff} < \textit{minDiff}$,那么更新 $\textit{minDiff}$ 为 $\textit{diff}$,更新答案为 $[[x,y]]$。
- 如果 $\textit{diff} = \textit{minDiff}$,那么把 $[x,y]$ 添加到答案中。
class Solution:
def minimumAbsDifference(self, arr: List[int]) -> List[List[int]]:
arr.sort()
min_diff = inf
ans = []
for x, y in pairwise(arr):
diff = y - x
if diff < min_diff:
min_diff = diff
ans = [[x, y]]
elif diff == min_diff:
ans.append([x, y])
return ans
class Solution {
public List<List<Integer>> minimumAbsDifference(int[] arr) {
Arrays.sort(arr);
int minDiff = Integer.MAX_VALUE;
List<List<Integer>> ans = new ArrayList<>();
for (int i = 1; i < arr.length; i++) {
int x = arr[i - 1];
int y = arr[i];
int diff = y - x;
if (diff < minDiff) {
minDiff = diff;
ans.clear();
ans.add(List.of(x, y));
} else if (diff == minDiff) {
ans.add(List.of(x, y));
}
}
return ans;
}
}
class Solution {
public:
vector<vector<int>> minimumAbsDifference(vector<int>& arr) {
ranges::sort(arr);
int min_diff = INT_MAX;
vector<vector<int>> ans;
for (int i = 1; i < arr.size(); i++) {
int x = arr[i - 1], y = arr[i];
int diff = y - x;
if (diff < min_diff) {
min_diff = diff;
ans = {{x, y}};
} else if (diff == min_diff) {
ans.push_back({x, y});
}
}
return ans;
}
};
int cmp(const void* a, const void* b) {
return *(int*)a - *(int*)b;
}
int** minimumAbsDifference(int* arr, int arrSize, int* returnSize, int** returnColumnSizes) {
qsort(arr, arrSize, sizeof(int), cmp);
int min_diff = INT_MAX;
int** ans = malloc((arrSize - 1) * sizeof(int*));
int k = 0;
for (int i = 0; i + 1 < arrSize; i++) {
int x = arr[i];
int y = arr[i + 1];
int diff = y - x;
if (diff < min_diff) {
min_diff = diff;
k = 0;
}
if (diff <= min_diff) {
ans[k] = malloc(2 * sizeof(int));
ans[k][0] = x;
ans[k][1] = y;
k++;
}
}
*returnSize = k;
*returnColumnSizes = malloc(k * sizeof(int));
for (int i = 0; i < k; i++) {
(*returnColumnSizes)[i] = 2;
}
return ans;
}
func minimumAbsDifference(arr []int) (ans [][]int) {
slices.Sort(arr)
minDiff := math.MaxInt
for i, x := range arr[:len(arr)-1] {
y := arr[i+1]
diff := y - x
if diff < minDiff {
minDiff = diff
ans = [][]int{{x, y}}
} else if diff == minDiff {
ans = append(ans, []int{x, y})
}
}
return
}
var minimumAbsDifference = function(arr) {
arr.sort((a, b) => a - b);
const ans = [];
let minDiff = Infinity;
for (let i = 1; i < arr.length; i++) {
const x = arr[i - 1], y = arr[i];
const diff = y - x;
if (diff < minDiff) {
minDiff = diff;
ans.length = 0;
ans.push([x, y]);
} else if (diff === minDiff) {
ans.push([x, y]);
}
}
return ans;
};
impl Solution {
pub fn minimum_abs_difference(mut arr: Vec<i32>) -> Vec<Vec<i32>> {
arr.sort_unstable();
let mut min_diff = i32::MAX;
let mut ans = vec![];
for i in 1..arr.len() {
let x = arr[i - 1];
let y = arr[i];
let diff = y - x;
if diff < min_diff {
min_diff = diff;
ans.clear();
ans.push(vec![x, y]);
} else if diff == min_diff {
ans.push(vec![x, y]);
}
}
ans
}
}
复杂度分析
- 时间复杂度:$\mathcal{O}(n\log n)$,其中 $n$ 是 $\textit{arr}$ 的长度。瓶颈在排序上。
- 空间复杂度:$\mathcal{O}(1)$。返回值和排序的栈开销不计入。
分类题单
- 滑动窗口与双指针(定长/不定长/单序列/双序列/三指针/分组循环)
- 二分算法(二分答案/最小化最大值/最大化最小值/第K小)
- 单调栈(基础/矩形面积/贡献法/最小字典序)
- 网格图(DFS/BFS/综合应用)
- 位运算(基础/性质/拆位/试填/恒等式/思维)
- 图论算法(DFS/BFS/拓扑排序/基环树/最短路/最小生成树/网络流)
- 动态规划(入门/背包/划分/状态机/区间/状压/数位/数据结构优化/树形/博弈/概率期望)
- 常用数据结构(前缀和/差分/栈/队列/堆/字典树/并查集/树状数组/线段树)
- 数学算法(数论/组合/概率期望/博弈/计算几何/随机算法)
- 贪心与思维(基本贪心策略/反悔/区间/字典序/数学/思维/脑筋急转弯/构造)
- 链表、树与回溯(前后指针/快慢指针/DFS/BFS/直径/LCA)
- 字符串(KMP/Z函数/Manacher/字符串哈希/AC自动机/后缀数组/子序列自动机)
欢迎关注 B站@灵茶山艾府