// push down heap element data[index] in data[0..end] // array index start with 0 instead of 1 in CLR void max_heapify(vector<int> &data, int end, int curr_root) { int left_child_idx = curr_root * 2 + 1; int right_child_idx = curr_root * 2 + 2; int largest_index = curr_root; if(left_child_idx <= end && data[largest_index] < data[left_child_idx]) largest_index = left_child_idx; if(right_child_idx <= end && data[largest_index] < data[right_child_idx]) largest_index = right_child_idx; if(largest_index != curr_root) { std::swap(data[largest_index], data[curr_root]); max_heapify(data, end, largest_index); } }
/*
index >= size/2 are all leaves, because the child index (size/2 * 2 + 1) is out of bound,
so we can start build heap from the leave nodes
*/
// make array data[0..end] into a heap void build_heap(vector<int> &data, int end) { for(int idx = data.size() / 2; idx >= 0; --idx) max_heapify(data, end, idx); } // 0) heap_size = data.size() - 1; // 1) make array data[0..heap_size] into a heap // 2) swap data[0] (max element) to the end // 3) decrease heap size by 1 // 4) goto 1) void heapsort(vector<int> &data) { int heap_size = data.size() - 1; build_heap(data, heap_size); for(int idx = data.size()-1; idx >= 1; --idx) { std::swap(data[0], data[idx]); heap_size--; max_heapify(data, heap_size, 0); } }