HashMap原理,hashmap的實現原理

來源:互聯網
上載者:User

HashMap原理,hashmap的實現原理
1.HashMap的資料結構

  數組的特點是:定址容易,插入和刪除困難;而鏈表的特點是:定址困難,插入和刪除容易。那麼我們能不能綜合兩者的特性,做出一種定址容易,插入刪除也容易的資料結構?答案是肯定的,這就是我們要提起的雜湊表,雜湊表有多種不同的實現方法,我接下來解釋的是最常用的一種方法—— 拉鏈法,我們可以理解為“鏈表的數組” ,

  從我們可以發現雜湊表是由數組+鏈表組成的,一個長度為16的數組中,每個元素儲存的是一個鏈表的頭結點。那麼這些元素是按照什麼樣的規則儲存到數組中呢。一般情況是通過hash(key)%len獲得,也就是元素的key的雜湊值對數組長度模數得到。比如上述雜湊表中,12%16=12,28%16=12,108%16=12,140%16=12。所以12、28、108以及140都儲存在數組下標為12的位置。

  HashMap其實也是一個線性數組實現的,所以可以理解為其儲存資料的容器就是一個線性數組。這可能讓我們很不解,一個線性數組怎麼實現按索引值對來存取資料呢?這裡HashMap有做一些處理。

  1.首先HashMap裡面實現一個靜態內部類Entry,其重要的屬性有 key , value, next,從屬性key,value我們就能很明顯的看出來Entry就是HashMap索引值對實現的一個基礎bean,我們上面說到HashMap的基礎就是一個線性數組,這個數組就是Entry[],Map裡面的內容都儲存在Entry[]裡面。

2.HashMap的存取實現

     既然是線性數組,為什麼能隨機存取?這裡HashMap用了一個小演算法,大致是這樣實現:

1 //儲存時:2 int hash = key.hashCode();// 這個hashCode方法這裡不詳述,只要理解每個key的hash是一個固定的int值3 int index = hash % Entry[].length;4 Entry[index] = value;5 6 //取值時:7 int hash = key.hashCode();8 int index = hash % Entry[].length;9 return Entry[index];

到這裡我們輕鬆的理解了HashMap通過索引值對實現存取的基本原理

    3.疑問:如果兩個key通過hash%Entry[].length得到的index相同,會不會有覆蓋的危險?

  這裡HashMap裡面用到鏈式資料結構的一個概念。上面我們提到過Entry類裡面有一個next屬性,作用是指向下一個Entry。打個比方, 第一個索引值對A進來,通過計算其key的hash得到的index=0,記做:Entry[0] = A。一會後又進來一個索引值對B,通過計算其index也等於0,現在怎麼辦?HashMap會這樣做:B.next = A,Entry[0] = B,如果又進來C,index也等於0,那麼C.next = B,Entry[0] = C;這樣我們發現index=0的地方其實存取了A,B,C三個索引值對,他們通過next這個屬性連結在一起。所以疑問不用擔心。也就是說數組中儲存的是最後插入的元素。到這裡為止,HashMap的大致實現,我們應該已經清楚了。

  當然HashMap裡面也包含一些最佳化方面的實現,這裡也說一下。比如:Entry[]的長度一定後,隨著map裡面資料的越來越長,這樣同一個index的鏈就會很長,會不會影響效能?HashMap裡面設定一個因素(也稱為因子),隨著map的size越來越大,Entry[]會以一定的規則加長長度。

3.解決hash衝突的辦法

Java中hashmap的解決辦法就是採用的鏈地址法。

4.實現自己的HashMap

  MapTest.java

1 package General; 2 import java.util.*; 3 public class MapTest { 4 public static void main(String[] args){ 5 Map<String,Employee> staff=new HashMap<>(); 6 staff.put("144-25-5456",new Employee("Amy Lee")); 7 staff.put("567-24-2456",new Employee("Harry Hacker")); 8 staff.put("157-62-7935",new Employee("Gary Cooper")); 9 staff.put("465-62-5537",new Employee("Francesca Cruz"));10 11 //print all entries12 System.out.println(staff);13 14 //remove an entry15 staff.remove("567-24-2456");16 17 //replace an entry18 staff.put("456-62-5527",new Employee("Francesca Miller"));19 20 //look up a value21 System.out.println(staff.get("157-62-7935"));22 23 //iterate through all entries24 for(Map.Entry<String, Employee>entry:staff.entrySet()){25 String key=entry.getKey();26 Employee value=entry.getValue();27 System.out.println("Key="+key+", value="+value);28 }29 }30 }31 class Employee{32 private String name;33 public Employee(String n){34 name=n;35 }36 public String getName(){37 return name;38 }39 }View Code1 {157-62-7935=Amy Lee, 2 567-24-2456=Harry Hacker, 3 144-25-5456=Gary Cooper, 4 465-62-5537=Francesca Cruz}5 Francesca Miller6 Key=157-62-7935, value=Amy Lee7 Key=144-25-5456, value=Harry Hacker8 Key=465-62-5537, value=Gary Cooper9 Key=456-62-5527, value=Francesca CruzView Code

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.