斜堆和左式堆的差别是,左式堆仅仅有在右孩子的npl大于左孩子的npl时,交换两个孩子的位置。而斜堆是不管怎样都交换:

package com.iflytek.heap;

/**
 * 斜堆
 * @author fgtian
 *
 */
public class SkewHeap {
	public static class HeapNode {
		int mValue;
		int mNpl = 0;
		HeapNode mLeftChild;
		HeapNode mRightChild;
	}
	
	HeapNode mRoot = null;
	
	public void insert(int value) {
		HeapNode node = new HeapNode();
		node.mValue = value;
		merge(node);
	}
	
	public int delMin() {
		if (null == mRoot) {
			throw new NullPointerException("null == mRoot");
		}
		int value = mRoot.mValue;
		mRoot = merge(mRoot.mLeftChild, mRoot.mRightChild);
		return value;
	}
	
	public void merge(HeapNode node) {
		mRoot = merge(mRoot, node);
		HeapNode l = mRoot.mLeftChild;
		mRoot.mLeftChild = mRoot.mRightChild;
		mRoot.mRightChild = l;
		mRoot.mNpl = Math.min(npl(mRoot.mLeftChild), npl(mRoot.mRightChild)) + 1;
	}
	
	public void merge(SkewHeap heap) {
		merge(heap.mRoot);
	}
	
	public static int npl(HeapNode h) {
		if (null == h) {
			return -1;
		}
		return h.mNpl;
	}
	
	public static HeapNode merge(HeapNode h1, HeapNode h2) {
		if (null == h1) {
			return h2;
		} else if (h2 == null) {
			return h1;
		} else { // 两个都不是null
			int v1 = h1.mValue;
			int v2 = h2.mValue;
			if (v1 <= v2) { // 拿他的
				h1.mRightChild = merge(h1.mRightChild, h2);
				HeapNode node = h1.mLeftChild;
				h1.mLeftChild = h1.mRightChild;
				h1.mRightChild = node;
				h1.mNpl = Math.min(npl(h1.mLeftChild), npl(h1.mRightChild)) + 1;
				return h1;
			} else {
				h2.mRightChild = merge(h1, h2.mRightChild);
				HeapNode node = h2.mLeftChild;
				h2.mLeftChild = h2.mRightChild;
				h2.mRightChild = node;
				h2.mNpl = Math.min(npl(h2.mLeftChild), npl(h2.mRightChild)) + 1;
				return h2;
			}
		}
	}
}