符号表
定义:
- 符号表最主要的目的就是将一个键和一个值联系起来,符号表能够将存储的数据元素是一个键和一个值共同组成的键值对数据,我们可以根据键来查找对应的值。
符号表中,键具有唯一性。
data:image/s3,"s3://crabby-images/d4a3a/d4a3a3b5d73e9319f09179a35de2e6831eddad7f" alt="在这里插入图片描述"
使用场景:
符号表在实际生活中的使用场景是非常广泛的,见下表: data:image/s3,"s3://crabby-images/a1278/a12785b7747695b75c44b6ea4f4352e7c443bf3f" alt="在这里插入图片描述"
链表实现
符号表API设计:
结点类: data:image/s3,"s3://crabby-images/40588/405882fa6b141f426238ca194ffef516feb204d8" alt="在这里插入图片描述" 符号表: data:image/s3,"s3://crabby-images/4c1d9/4c1d9f79d6a01db53991806fc6341655dc2ef649" alt="在这里插入图片描述"
代码实现:
public class SymbolTable<Key,Value> {
private Node head;
private int N;
public SymbolTable() {
this.head = new Node(null,null,null);
this.N=0;
}
public int size(){
return N;
}
public void put(Key key,Value value){
Node n = head;
while(n.next!=null){
n = n.next;
if (n.key.equals(key)){
n.value = value;
return;
}
}
Node newNode = new Node(key, value, null);
Node oldFirst = head.next;
newNode.next = oldFirst;
head.next = newNode;
N++;
}
public void delete(Key key){
Node n = head;
while(n.next!=null){
if (n.next.key.equals(key)){
n.next = n.next.next;
N--;
return;
}
n = n.next;
}
}
public Value get(Key key){
Node n = head;
while(n.next!=null){
n = n.next;
if (n.key.equals(key)){
return n.value;
}
}
return null;
}
private class Node{
public Key key;
public Value value;
public Node next;
public Node(Key key, Value value, Node next) {
this.key = key;
this.value = value;
this.next = next;
}
}
}
测试:
public class SymbolTableTest {
public static void main(String[] args) {
SymbolTable<Integer, String> symbolTable = new SymbolTable<>();
symbolTable.put(1,"乔峰");
symbolTable.put(2,"虚竹");
symbolTable.put(3,"段誉");
System.out.println("插入完毕后,元素的个数为:"+symbolTable.size());
symbolTable.put(2, "慕容复");
System.out.println("替换完毕后的元素的个数为:"+symbolTable.size());
System.out.println("替换完毕后,键2对应的值为:"+symbolTable.get(2));
symbolTable.delete(2);
System.out.println("删除完毕后,元素的个数:"+symbolTable.size());
}
}
data:image/s3,"s3://crabby-images/f2f12/f2f12b695ca4d56a11069481e4dcef33939cfae1" alt="在这里插入图片描述"
有序符号表
- 刚才实现的符号表,我们可以称之为无序符号表,因为在插入的时候,并没有考虑键值对的顺序,而在实际生活中,有时候我们需要根据键的大小进行排序,插入数据时要考虑顺序,那么接下来我们就实现一下有序符号表。
data:image/s3,"s3://crabby-images/d5ee2/d5ee2dd4d6e6678fc38d75b329acd193313280ef" alt="在这里插入图片描述"
有序链表实现:
public class OrderSymbolTable<Key extends Comparable<Key>,Value> {
private Node head;
private int N;
private class Node{
public Key key;
public Value value;
public Node next;
public Node(Key key, Value value, Node next) {
this.key = key;
this.value = value;
this.next = next;
}
}
public OrderSymbolTable() {
this.head = new Node(null,null,null);
this.N=0;
}
public int size(){
return N;
}
public void put(Key key,Value value){
Node curr = head.next;
Node pre = head;
while(curr!=null && key.compareTo(curr.key)>0){
pre = curr;
curr = curr.next;
}
if (curr!=null && key.compareTo(curr.key)==0){
curr.value = value;
return;
}
Node newNode = new Node(key, value, curr);
pre.next = newNode;
N++;
}
public void delete(Key key){
Node n = head;
while(n.next!=null){
if (n.next.key.equals(key)){
n.next = n.next.next;
N--;
return;
}
n = n.next;
}
}
public Value get(Key key){
Node n = head;
while(n.next!=null){
n = n.next;
if (n.key.equals(key)){
return n.value;
}
}
return null;
}
}
debug测试:
data:image/s3,"s3://crabby-images/8ad88/8ad88997329e96424b79854017fe54a3f15e671e" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/66ca6/66ca641b88676ec338c9d4d9631d7b838e27fe45" alt="在这里插入图片描述"
数组二分查找实现:
使用一对平行数组,一个存储键一个存储值。
- 二分查找的思想是在内部维护一个按照key排好序的二维数组,每一次查找的时候,跟中间元素进行比较,如果该元素小,则继续左半部分递归查找,否则继续右半部分递归查找。整个实现代码如下:
- 二分查找的 rank() 方法至关重要,当键在表中时,它能够知道该键的位置;当键不在表中时,它也能知道在何处插入新键。
data:image/s3,"s3://crabby-images/edeb3/edeb3e39757505fa8da0a8c4436c9e8ec179d5ec" alt="在这里插入图片描述"
public class SymbolTable<K extends Comparable<K>,V> {
private K[] keys;
private V[] values;
public int size;
private static final int initSize = 10;
public SymbolTable(){
this(initSize);
}
public SymbolTable(int capacity) {
keys = (K[]) new Comparable[capacity];
values = (V[]) new Object[capacity];
}
public V get(K k){
if(isEmpty()){
return null;
}
int i = rank(k);
if(i < size && keys[i].compareTo(k) == 0){
return values[i];
}
return null;
}
public void put(K k,V v){
int i = rank(k);
if(i < size && keys[i].compareTo(k)==0){
values[i] = v;
return;
}
for(int j = size ;j>i ; j--){
keys[j] = keys[j - 1];
values[j] = values[j - 1];
}
keys[i] = k;
values[i] = v;
size ++;
}
public boolean isEmpty(){
return size == 0;
}
public int rank(K k){
int low = 0;
int high = size-1;
while(low <= high){
int mid = low + (high - low) /2;
int cmd = k.compareTo(keys[mid]);
if(cmd == 0){
return mid;
}else if(cmd < 0){
high = mid - 1;
}else {
low = mid + 1;
}
}
return low;
}
}
debug测试:
data:image/s3,"s3://crabby-images/ad81f/ad81f363d03f2e02bd806d10f5ed8b778da9de8c" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/f639f/f639f1974460c99dec3b5aabcf10f055227e3718" alt="在这里插入图片描述"
总结:
本文介绍了符号表这一抽象数据结构,然后介绍了两种基本实现:基于无序链表的实现和基于有序数组的实现,两种实现的时间复杂度如下:
data:image/s3,"s3://crabby-images/5c2f0/5c2f0725c0141f4de7c481a4b492eea8fa497dab" alt="在这里插入图片描述"
无序链表实现:
- 插入的时候先要查找,如果存在则更新value,查找的时候需要从链表头进行查找,所以插入和查找的平均时间复杂度均为O(n)
数组二分查找:
- 采用二分查找只需要最多 logN+1次的比较即可找到对应元素,所以查找效率比较高。
- 但是对于插入元素来说,每一次插入不存在的元素,需要将该元素放到指定的位置,然后,将他后面的元素依次后移,所以平均时间复杂度O(n),对于插入来说效率仍然比较低。
使用有序数组的二分查找法提高了符号表的查找速度,但是插入效率仍旧没有得到提高,而且在要维护数组有序,还需要进行排序操作。这两种实现方式简单直观,但是无法同时达到较高查找和插入效率。
本文只是一个引子,后面的系列文章将会介绍二叉查找树,平衡查找树以及哈希表。
|