Hashtable 的实现原理

Hashtable 的实现原理

【IT168 技术文档】在仔细分析源代码之前,我们来看看Hashtable提供的一些接口方法。

public
 
int
 size();

public
 
boolean
 isEmpty() ;

public
 
synchronized
 Enumeration keys();

public
 
synchronized
 Enumeration elements();

public
 
synchronized
 
boolean
 contains(Object value) ;

public
 
synchronized
 
boolean
 containsKey(Object key);

public
 
synchronized
 Object get(Object key);

public
 
synchronized
 Object put(Object key, Object value) ;

public
 
synchronized
 Object remove(Object key);

public
 
synchronized
 
void
 clear() ;

public
 
synchronized
 
void
 clear() ;

 

   上面的方法我就不一一介绍了,具体的用法也是很简单,相对大家对此也比较熟悉了。

Hashtable的用法

Hashtable 有2个构造函数

public
 Hashtable(
int
 initialCapacity); 
//
指定容量大小


public
 Hashtable() {

        

this
(
11
); 
//
默认的容量是11,为什么是11,而不是10呢?


}

Demo1

Hashtable sTable 
=
 
new
 Hashtable();
  sTable.put(


wuhua

,

wuhua

);
  sTable.remove(


wuhua

);
  sTable.clear();

上面是简单的用法。

Hashtable源代码解读

  在了解源代码之前,我们先来了解下一些java不常用的关键字。

  • transient

  当串行化某个对象时,如果该对象的某个变量是transient,那么这个变量不会被串行化进去。也就是说,假设某个类的成员变量是transient,那么当通过ObjectOutputStream把这个类的某个实例保存到磁盘上时,实际上transient变量的值是不会保存的。因为当从磁盘中读出这个对象的时候,对象的该变量会没有被赋值。 另外这篇文章还提到,当从磁盘中读出某个类的实例时,实际上并不会执行这个类的构造函数,而是读取这个类的实例的状态,并且把这个状态付给这个类的对象。

引用地址:http://blog.chinaunix.net/u/22516/showart.php?id=380029

  Transient 这个关键字很重要,来看下源代码里面有几处是用到这个关键字的。

private
 
transient
 HashtableEntry table[];

private
 
transient
 
int
 count;    

  源代码中只有上面两个字段的定义是用到的,但是这两个字段是用于存储Hashtable的容器字段,因此可以说Hashtable是不允许序列化的。

  • 内部类

  Hashtable有2个内部类
  HashtableEntry — 用于存放key-value,nextElement的类。

class
 HashtableEntry {

    

int
 hash;
    Object key;
    Object value;
    HashtableEntry next;
}

 

  HashtableEnumerator 遍历的枚举类。

 

class
 HashtableEnumerator 
implements
 Enumeration {

        

boolean
 keys;
        

int
 index;
        HashtableEntry table[];
        HashtableEntry entry;

        HashtableEnumerator(HashtableEntry table[], 

boolean
 keys) {

            

this
.table 
=
 table;
            

this
.keys 
=
 keys;
            

this
.index 
=
 table.length;
        }

        

public
 
boolean
 hasMoreElements() {

            

if
 (entry 
!=
 
null
) {

                

return
 
true
;
            }
            

while
 (index

 
>
 
0
) {

                

if
 ((entry 
=
 table[index]) 
!=
 
null
) {

                    

return
 
true
;
                }
            }
            

return
 
false
;
        }

        

public
 Object nextElement() {

            

if
 (entry 
==
 
null
) {

                

while
 ((index

 
>
 
0

&&
 ((entry 
=
 table[index]) 
==
 
null
));
            }
            

if
 (entry 
!=
 
null
) {

                HashtableEntry e 

=
 entry;
                entry 

=
 e.next;
                

return
 keys 
?
 e.key : e.value;
            }
            

throw
 
new
 NoSuchElementException(

/*
 #ifdef VERBOSE_EXCEPTIONS 
*/


//
/ skipped                       “HashtableEnumerator”

/*

 #endif 
*/

            );
        }
    }

  代码写的是相当的简介。有一些比较技巧性的用法也是相当的不错,比如:

if
 (entry 
==
 
null
) {

     

while
 ((index

 
>
 
0

&&
 ((entry 
=
 table[index]) 
==
 
null
));
} 这段写的是相当的好,可见作者的功力,循环变量table
  

while
 (index

 
>
 
0

   

//
循环变量,查看是否有下一个元素,


       
if
 ((entry 
=
 table[index]) 
!=
 
null
) {

                    

return
 
true
;
       }
   }

  了解了Hashtable的数据存放格式,我们看看存放的关键逻辑
在put,remove,get方法中存在。

int
 index 
=
 (hash 
&
 
0x7FFFFFFF

%
 tab.length;

这样的函数,这个函数的意义上,根据散列值来获取对象的存储位置。
来欣赏下代码片段:

public
 
synchronized
 Object get(Object key) {

        HashtableEntry tab[] 

=
 table;
        

int
 hash 
=
 key.hashCode();
        

int
 index 
=
 (hash 
&
 
0x7FFFFFFF

%
 tab.length;
        

for
 (HashtableEntry e 
=
 tab[index] ; e 
!=
 
null
 ; e 
=
 e.next) {

            

if
 ((e.hash 
==
 hash) 
&&
 e.key.equals(key)) {

                

return
 e.value;
            }
        }
        

return
 
null
;
}

从上面的代码可以分析出。首先获取key的散列值,并且根据散列值进行key的Index定位
这里存在同一个index多个HashtableEntry 存在,所以才会有了next的变量,next就是存放相同位置不同key的实体。

下面再来看看Hashtable里面一个扩充容器的算法。

 

protected
 
void
 rehash() {

        

int
 oldCapacity 
=
 table.length;
        HashtableEntry oldTable[] 

=
 table;

        

int
 newCapacity 
=
 oldCapacity 
*
 
2
 
+
 
1
;
        HashtableEntry newTable[] 

=
 
new
 HashtableEntry[newCapacity];

        threshold 

=
 (
int
)((newCapacity 
*
 loadFactorPercent) 
/
 
100
);
        table 

=
 newTable;

        

for
 (
int
 i 
=
 oldCapacity ; i

 
>
 
0
 ;) {

        

//
循环遍历oldTable的对应的实体,并且遍历对应的实体的没一个对象,进行
        

//
重新分配index,再进行保存


            
for
 (HashtableEntry old 
=
 oldTable[i] ; old 
!=
 
null
 ; ) {

                HashtableEntry e 

=
 old;
                old 

=
 old.next;

                

int
 index 
=
 (e.hash 
&
 
0x7FFFFFFF

%
 newCapacity;
                e.next 

=
 newTable[index]; 
//
e 的next 指向当前索引


                newTable[index] 
=
 e;  
//
 


            }
        }
}

 

for
 (HashtableEntry old 
=
 oldTable[i] ; old 
!=
 
null
 ; ) 
这段的用法很奇怪,我以前没有使用过,上面的代码等同于
HashtableEntry old 

=
 oldTable[i];
Whild(old 

!=
 
null
){

………………………….
………………………….
}
不过个人比较习惯使用第2种方式

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/100466.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • nginx系列之一:nginx入门

    nginx系列之一:nginx入门一、nginx功能介绍Nginx因为它的稳定性、丰富的模块库、灵活的配置和低系统资源的消耗而闻名.业界一致认为它是Apache2.2+mod_proxy_balancer的轻量级代替者,不仅是因为响应静态页面的速度非常快,而且它的模块数量达到Apache的近2/3。对proxy和rewrite模块的支持很彻底,还支持mod_fcgi、ssl、vhosts,适合用来做mongrelclust…

    2022年6月2日
    31
  • 树莓派4b基础入门「建议收藏」

    树莓派4b基础入门「建议收藏」目录一、树莓派百科知识二、树莓派4B图解及配件选择三、如何烧录系统?四、树莓派开机连接五、常见警示标志和故障排除六、格式化TF卡七、系统备份与恢复八、无线WiFi上网配置九、系统汉化教程十、键盘布局设置十一、树莓派扩展分区十二、开启SSH的4种方法十三、开启VNC的3种方法十四、Windows远程桌面连接十五、获取IP和MAC地址十六、设置静态IP十七、常见问题一、树莓派百科知识树莓派(RaspberryPi)是一款基于ARM的微型电脑主板,旨为学生计算机编程教育而设计,其系统基于Linux,由注册于

    2022年6月11日
    123
  • java 新建项目_java怎么新建项目?java新建项目实操案例

    java 新建项目_java怎么新建项目?java新建项目实操案例java新建项目是学习java最基础的实操了,最近有小伙伴想知道java怎么新建项目?那么下面我们就来给大家讲解一下java新建项目的方法。1、选择“file(文件)”|“new(新建)”|“JavaProject(Java项目)”命令,打开“NewJavaProject(新建Java项目)”对话框。2、设置“Projectname(项目名)”为HelloJava,选中“Usedefau…

    2022年7月7日
    31
  • 眼下最好的JSP分页技术

    眼下最好的JSP分页技术

    2021年11月13日
    42
  • Ink笔记_ink correction

    Ink笔记_ink correction最近想要复刻一下稚晖君的小卡片,因此来学习一下。1.ST25DV作为NFC的PHY通过I2C总线和STM32通信,主要作用有两个:能量采集以及NFC通信。注意,ST25DV只是负责和手机进行NFC通信,而不负责IC卡的读写功能,因为ST25DV只支持ISO15693的RFID协议,而我们常用的IC卡(M1卡)是ISO14443协议的,所以并不能直接使用这颗芯片进行IC卡模拟。2.IC卡的模拟功能这一版中实现得比较简单,就是直接集成了多颗UID芯片(很便宜,1~2元一片),然后和ST25DV共用N

    2025年6月23日
    1
  • 公网远程开机(唤醒家庭PC)

    公网远程开机(唤醒家庭PC)一、背景前一段搞使用seafile搞了私有云盘,不过不需要的时候开着电脑好像有点浪费,所以就开始了通过公网开机的道路二、关键性问题问题要说在前面,常规性的东西百度一般配置都可以搞定,如果各种面向百度的配置都已经尝试,请注意如下问题。. 一定在被开机的直连路由设备上进行MAC与IP地址绑定(原理略)。局域网直接子网内广播发送可以不用绑定。 被开机的系统需要安装对应的网卡驱动(实验CentOS7是有问题的,windows用驱动精灵安装下网卡驱动搞定)三、通过互联网公网远程开机一般性步骤按照常

    2022年5月30日
    66

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号