mirror of
https://github.com/posborne/rust-pstree.git
synced 2025-01-14 00:06:14 +01:00
Use hashmap for O(n) instead of O(n^2) tree building from records
This commit is contained in:
parent
5823d36e84
commit
8d9bf2cefb
1 changed files with 35 additions and 13 deletions
46
pstree.rs
46
pstree.rs
|
@ -29,12 +29,14 @@
|
||||||
|
|
||||||
#![feature(old_io)]
|
#![feature(old_io)]
|
||||||
#![feature(old_path)]
|
#![feature(old_path)]
|
||||||
|
#![feature(std_misc)] // hash_map::Entry
|
||||||
|
|
||||||
use std::old_io::fs::PathExtensions;
|
use std::old_io::fs::PathExtensions;
|
||||||
use std::old_io::fs;
|
use std::old_io::fs;
|
||||||
use std::old_io::File;
|
use std::old_io::File;
|
||||||
use std::old_io::BufferedReader;
|
use std::old_io::BufferedReader;
|
||||||
|
use std::collections::hash_map::Entry::{Occupied, Vacant};
|
||||||
|
use std::collections::HashMap;
|
||||||
|
|
||||||
#[derive(Clone,Debug)]
|
#[derive(Clone,Debug)]
|
||||||
struct ProcessRecord {
|
struct ProcessRecord {
|
||||||
|
@ -110,20 +112,40 @@ fn get_process_records() -> Vec<ProcessRecord> {
|
||||||
}).collect()
|
}).collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn populate_node(node : &mut ProcessTreeNode, records: &Vec<ProcessRecord>) {
|
fn populate_node_helper(node: &mut ProcessTreeNode, pid_map: &HashMap<i32, &ProcessRecord>, ppid_map: &HashMap<i32, Vec<i32>>) {
|
||||||
// populate the node by finding its children... recursively
|
|
||||||
let pid = node.record.pid; // avoid binding node as immutable in closure
|
let pid = node.record.pid; // avoid binding node as immutable in closure
|
||||||
node.children.extend(
|
let child_nodes = &mut node.children;
|
||||||
records.iter().filter_map(|record| {
|
match ppid_map.get(&pid) {
|
||||||
if record.ppid == pid {
|
Some(children) => {
|
||||||
|
child_nodes.extend(children.iter().map(|child_pid| {
|
||||||
|
let record = pid_map[*child_pid];
|
||||||
let mut child = ProcessTreeNode::new(record);
|
let mut child = ProcessTreeNode::new(record);
|
||||||
populate_node(&mut child, records);
|
populate_node_helper(&mut child, pid_map, ppid_map);
|
||||||
Some(child)
|
child
|
||||||
} else {
|
}));
|
||||||
None
|
},
|
||||||
|
None => {},
|
||||||
}
|
}
|
||||||
})
|
}
|
||||||
)
|
|
||||||
|
fn populate_node(node : &mut ProcessTreeNode, records: &Vec<ProcessRecord>) {
|
||||||
|
// O(n): build a mapping of pids to vectors of children. That is, each
|
||||||
|
// key is a pid and its value is a vector of the whose parent pid is the key
|
||||||
|
let mut ppid_map : HashMap<i32, Vec<i32>> = HashMap::new();
|
||||||
|
let mut pid_map : HashMap<i32, &ProcessRecord> = HashMap::new();
|
||||||
|
for record in records.iter() {
|
||||||
|
// entry returns either a vacant or occupied entry. If vacant,
|
||||||
|
// we insert a new vector with this records pid. If occupied,
|
||||||
|
// we push this record's pid onto the vec
|
||||||
|
pid_map.insert(record.pid, record);
|
||||||
|
match ppid_map.entry(record.ppid) {
|
||||||
|
Vacant(entry) => { entry.insert(vec![record.pid]); },
|
||||||
|
Occupied(mut entry) => { entry.get_mut().push(record.pid); },
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// With the data structures built, it is off to the races
|
||||||
|
populate_node_helper(node, &pid_map, &ppid_map);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn build_process_tree() -> ProcessTree {
|
fn build_process_tree() -> ProcessTree {
|
||||||
|
|
Loading…
Reference in a new issue