- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
java解法
给定一个以数组形式表示的完全二叉树(从索引 1 开始表示节点,-1 表示空节点),找到所有叶子节点中值最小的叶子节点。
输出从根节点到该叶子节点的路径。
树的索引特性:
当前节点索引为 idx:
左子节点索引为 2 * idx。
右子节点索引为 2 * idx + 1。
根节点索引为 1。
如果索引超出数组范围,或者节点值为 -1,表示该节点为空。
实现方法:
使用递归深度优先搜索(DFS)遍历二叉树。
在递归过程中:
构建当前节点的路径。
如果是叶子节点(无左右子节点):
判断该节点是否是值最小的叶子节点。
如果是,更新结果路径。
如果不是叶子节点,继续递归其左右子节点。
递归结束后,返回存储的路径。
关键逻辑:
叶子节点判断:
当前节点的左右子节点不存在或为空时,认为是叶子节点。
路径更新:
使用 List 存储当前路径。
使用 List 存储结果路径。
最小值记录:
使用数组 minVal[0] 存储当前最小叶子节点的值。
时间复杂度:
遍历整个树,复杂度为 O(n),其中 n 是节点数
import java.util.*;
public class Main {
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
int[] tr = Arrays.stream(sc.nextLine().split(" ")).mapToInt(Integer::parseInt).toArray();
List<Integer> res = new ArrayList<>();
findMinLeafPath(tr, 1, res, new ArrayList<>(), new int[]{Integer.MAX_VALUE});
System.out.println(String.join(" ", res.stream().map(String::valueOf).toArray(String[]::new)));
}
private static void findMinLeafPath(int[] tr, int idx, List<Integer> res, List<Integer> curPath, int[] minVal) {
if (idx >= tr.length || tr[idx - 1] == -1) {
return;
}
curPath.add(tr[idx - 1]);
if (2 * idx >= tr.length ||
(tr[2 * idx - 1] == -1 &&
(2 * idx + 1 >= tr.length || tr[2 * idx] == -1))) {
if (tr[idx - 1] < minVal[0]) {
minVal[0] = tr[idx - 1];
res.clear();
res.addAll(new ArrayList<>(curPath));
}
} else {
findMinLeafPath(tr, 2 * idx, res, curPath, minVal);
findMinLeafPath(tr, 2 * idx + 1, res, curPath, minVal);
}
curPath.remove(curPath.size() - 1);
}
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
C++解法
更新中
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
C解法
更新中
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
JS解法
输入一个以数组形式表示的完全二叉树,其中 -1 表示空节点。
找出所有叶子节点中值最小的节点,并输出从根节点到该叶子节点的路径。
二叉树数组表示:
对于当前节点索引 index:
左子节点索引为 2 * index + 1。
右子节点索引为 2 * index + 2。
根节点索引为 0。
算法实现:
广度优先搜索 (BFS):
使用队列来遍历树,记录每个节点的索引和从根到该节点的路径。
每次从队列中取出一个节点:
如果是叶子节点(左右子节点不存在或为空),检查是否是当前最小叶子节点。
如果不是叶子节点,继续将其左右子节点加入队列。
路径记录:
每个节点在队列中记录其索引和从根到当前节点的路径。
更新最小叶子节点时,保存其路径。
时间复杂度:
遍历数组一次:O(n)。
总时间复杂度:O(n)
const readline = require("readline");
const rl = readline.createInterface({
input: process.stdin,
output: process.stdout,
});
rl.on("line", (line) => {
const arr = line.split(" ").map(Number);
let min = Infinity;
let path = [];
let queue = [{ index: 0, currentPath: [arr[0]] }];
while (queue.length > 0) {
const { index, currentPath } = queue.shift();
if (index >= arr.length || arr[index] === -1) continue;
const left = 2 * index + 1;
const right = 2 * index + 2;
if (left >= arr.length || arr[left] === -1) {
if (right >= arr.length || arr[right] === -1) {
if (arr[index] < min) {
min = arr[index];
path = currentPath;
}
}
}
if (left < arr.length) {
queue.push({ index: left, currentPath: [...currentPath, arr[left]] });
}
if (right < arr.length) {
queue.push({ index: right, currentPath: [...currentPath, arr[right]] });
}
}
console.log(path.join(" "));
});
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
注意:
如果发现代码有用例覆盖不到的情况,欢迎反馈!会在第一时间修正,更新。
解题不易,如对您有帮助,欢迎点赞/收藏
评论记录:
回复评论: