映射
映射是一种包含键值对的数据结构。
public interface Map<K, V> {
/**
* 添加
*
* @param key 键
* @param value 值
*/
void add(K key, V value);
/**
* 删除
*
* @param key 键
* @return 删除的元素
*/
V remove(K key);
/**
* 是否包含
*
* @param key 键
* @return 是否包含
*/
boolean contains(K key);
/**
* 获取key对应的值
*
* @param key 键
* @return 值
*/
V get(K key);
/**
* 设置
*
* @param key 键
* @param value 值
*/
void set(K key, V value);
/**
* 获取元素个数
*
* @return 元素个数
*/
int getSize();
/**
* 是否为空
*
* @return 是否为空
*/
boolean isEmpty();
}
基于链表的映射
public class LinkedListMap<K, V> implements Map<K, V> {
private class Node {
/**
* 见
*/
public K key;
/**
* 值
*/
public V value;
/**
* 下一个节点
*/
public Node next;
/**
* 构造函数
*
* @param key 键
* @param value 值
* @param next 下一个节点
*/
public Node(K key, V value, Node next) {
this.key = key;
this.value = value;
this.next = next;
}
/**
* 构造函数
*
* @param key 键
* @param value 值
*/
public Node(K key, V value) {
this(key, value, null);
}
/**
* 无参构造函数
*/
public Node() {
this(null, null, null);
}
@Override
public String toString() {
return key.toString() + " : " + value.toString();
}
}
/**
* 虚拟头节点
*/
private Node dummyHead;
/**
* 元素个数
*/
private int size;
/**
* 无参构造函数
*/
public LinkedListMap() {
dummyHead = new Node();
size = 0;
}
@Override
public int getSize() {
return size;
}
@Override
public boolean isEmpty() {
return size == 0;
}
/**
* 根据key查找节点
*
* @param key key
* @return 节点
*/
private Node getNode(K key) {
Node cur = dummyHead.next;
while (cur != null) {
if (cur.key.equals(key)) {
return cur;
}
cur = cur.next;
}
return null;
}
@Override
public boolean contains(K key) {
return getNode(key) != null;
}
@Override
public V get(K key) {
Node node = getNode(key);
return node == null ? null : node.value;
}
@Override
public void add(K key, V value) {
Node node = getNode(key);
if (node == null) {
dummyHead.next = new Node(key, value, dummyHead.next);
size++;
} else {
node.value = value;
}
}
@Override
public void set(K key, V newValue) {
Node node = getNode(key);
if (node == null) {
throw new IllegalArgumentException(key + " doesn't exist!");
}
node.value = newValue;
}
@Override
public V remove(K key) {
Node prev = dummyHead;
while (prev.next != null) {
if (prev.next.key.equals(key)) {
break;
}
prev = prev.next;
}
if (prev.next != null) {
Node delNode = prev.next;
prev.next = delNode.next;
delNode.next = null;
size--;
return delNode.value;
}
return null;
}
}
基于二分搜索树的映射
public class BSTMap<K extends Comparable<K>, V> implements Map<K, V> {
private class Node {
/**
* 键
*/
public K key;
/**
* 值
*/
public V value;
/**
* 左孩子节点
*/
public Node left;
/**
* 右孩子节点
*/
public Node right;
/**
* 构造函数
*
* @param key 键
* @param value 值
*/
public Node(K key, V value) {
this.key = key;
this.value = value;
left = null;
right = null;
}
}
/**
* 根节点
*/
private Node root;
/**
* 元素个数
*/
private int size;
public BSTMap() {
root = null;
size = 0;
}
@Override
public int getSize() {
return size;
}
@Override
public boolean isEmpty() {
return size == 0;
}
/**
* 向二分搜索树中添加新的元素(key, value)
*
* @param key 键
* @param value 值
*/
@Override
public void add(K key, V value) {
root = add(root, key, value);
}
/**
* 向以node为根的二分搜索树中插入元素(key, value),递归算法
*
* @param node 根节点
* @param key 键
* @param value 值
* @return 插入新节点后二分搜索树的根
*/
private Node add(Node node, K key, V value) {
if (node == null) {
size++;
return new Node(key, value);
}
if (key.compareTo(node.key) < 0) {
node.left = add(node.left, key, value);
} else if (key.compareTo(node.key) > 0) {
node.right = add(node.right, key, value);
} else {
// key.compareTo(node.key) == 0
node.value = value;
}
return node;
}
/**
* 返回以node为根节点的二分搜索树中,key所在的节点
*
* @param node 根节点
* @param key key
* @return 节点
*/
private Node getNode(Node node, K key) {
if (node == null) {
return null;
}
if (key.equals(node.key)) {
return node;
} else if (key.compareTo(node.key) < 0) {
return getNode(node.left, key);
} else {
return getNode(node.right, key);
}
}
@Override
public boolean contains(K key) {
return getNode(root, key) != null;
}
@Override
public V get(K key) {
Node node = getNode(root, key);
return node == null ? null : node.value;
}
@Override
public void set(K key, V newValue) {
Node node = getNode(root, key);
if (node == null) {
throw new IllegalArgumentException(key + " doesn't exist!");
}
node.value = newValue;
}
/**
* 返回以node为根的二分搜索树的最小值所在的节点
*
* @param node 根节点
* @return 最小值所在的节点
*/
private Node minimum(Node node) {
if (node.left == null) {
return node;
}
return minimum(node.left);
}
/**
* 删除掉以node为根的二分搜索树中的最小节点
*
* @param node 根节点
* @return 删除节点后新的二分搜索树的根
*/
private Node removeMin(Node node) {
if (node.left == null) {
Node rightNode = node.right;
node.right = null;
size--;
return rightNode;
}
node.left = removeMin(node.left);
return node;
}
/**
* 从二分搜索树中删除键为key的节点
*
* @param key 键
* @return 值
*/
@Override
public V remove(K key) {
Node node = getNode(root, key);
if (node != null) {
root = remove(root, key);
return node.value;
}
return null;
}
private Node remove(Node node, K key) {
if (node == null) {
return null;
}
if (key.compareTo(node.key) < 0) {
node.left = remove(node.left, key);
return node;
} else if (key.compareTo(node.key) > 0) {
node.right = remove(node.right, key);
return node;
} else {
// key.compareTo(node.key) == 0
// 待删除节点左子树为空的情况
if (node.left == null) {
Node rightNode = node.right;
node.right = null;
size--;
return rightNode;
}
// 待删除节点右子树为空的情况
if (node.right == null) {
Node leftNode = node.left;
node.left = null;
size--;
return leftNode;
}
// 待删除节点左右子树均不为空的情况
// 找到比待删除节点大的最小节点, 即待删除节点右子树的最小节点
// 用这个节点顶替待删除节点的位置
Node successor = minimum(node.right);
successor.right = removeMin(node.right);
successor.left = node.left;
node.left = node.right = null;
return successor;
}
}
}