欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java數(shù)據(jù)結(jié)構(gòu)之實現(xiàn)哈希表的分離鏈接法

 更新時間:2021年06月30日 11:35:11   作者:TanGBx  
今天給大家?guī)淼氖顷P(guān)于Java數(shù)據(jù)結(jié)構(gòu)的相關(guān)知識,文章圍繞著Java哈希表的分離鏈接法展開,文中有非常詳細的介紹及代碼示例,需要的朋友可以參考下

哈希表的分離鏈接法

原理

Hash Table可以看作是一種特殊的數(shù)組。他的原理基本上跟數(shù)組相同,給他一個數(shù)據(jù),經(jīng)過自己設(shè)置的哈希函數(shù)變換得到一個位置,并在這個位置當(dāng)中放置該數(shù)據(jù)。哦對了,他還有個名字叫散列

0 1
數(shù)據(jù)1 數(shù)據(jù)2

就像這個數(shù)組,0號位置放著數(shù)據(jù)1,1號位置放數(shù)據(jù)2
而我們的哈希表則是通過一個函數(shù)f(x) 把數(shù)據(jù)1變成0,把數(shù)據(jù)2變成1,然后在得到位置插入數(shù)據(jù)1和數(shù)據(jù)2。
非常重要的是哈希表的長度為素數(shù)最好??!
而且當(dāng)插入數(shù)據(jù)大于一半的時候我們要進行擴充?。?!

沖突問題產(chǎn)生

現(xiàn)在這個表就是2個數(shù)據(jù),所以不會產(chǎn)生什么沖突,但是若一個數(shù)據(jù)他通過f(x)計算得到的位置也是0呢?是不是就跟數(shù)據(jù)1產(chǎn)生了沖突,因為數(shù)據(jù)1已經(jīng)占據(jù)了這個位置,你無法進行插入操作。對不對。

所以我們該如何解決這個問題呢,誒,我們肯定是想兩個都可以插入是不是,就像一個炸串一樣把他串起來。如圖

在這里插入圖片描述

a b c就像一個炸串,而如何實現(xiàn)這個炸串就有多種方式。這里我們用線性表來實現(xiàn)

線性表實現(xiàn)

我們可以用java自帶的List ArrayList等表,這邊也給出單鏈表的實現(xiàn)方式。

public class MyArray<AnyType> {

我們首先得創(chuàng)建一個內(nèi)部類用來存放數(shù)據(jù),以及保存下個節(jié)點

 class ArrayNode<AnyType>{
     public AnyType data;
     public ArrayNode<AnyType> next;
     public ArrayNode(AnyType data){this(data,null);}
     private ArrayNode(AnyType data,ArrayNode<AnyType> next){
         this.data=data;
         this.next=next;
     }
 }//save type node;

設(shè)置我們這個線性表所需要的對象,例如size和一個頭節(jié)點,以及我們要進行初始化,判斷這個表是否為空等。

 private int theSize;//array list size
 private ArrayNode<AnyType> head; //head node every data behind it
 //init MyArray
 public MyArray(){doClear();}
 public void clear(){doClear();}
 private void doClear(){
     theSize=0;
     head=new ArrayNode<>(null);
 }
 //get size and is empty
 public int size(){return theSize;}
 public boolean isEmpty(){return theSize==0;}

接下來我們需要實現(xiàn)他的基本操作,是否包含,插入,獲得以及刪除。

   //contain
   public boolean contains(AnyType x){
       ArrayNode<AnyType> newNode=head;//get a new node=head
       while (newNode.next!=null) {
           newNode=newNode.next;
           if (newNode.data == x)
               return true;
       }
       return false;
   }
   //get the data in idx from array
   public AnyType get(int idx){return get(idx,head).data;}
   private ArrayNode<AnyType> get(int idx,ArrayNode<AnyType> node){
       if(idx<0||idx>size())
           throw new IndexOutOfBoundsException();//out of length
       ArrayNode<AnyType> newNode=node;
       //find start head.next
       for (int i = 0; i < idx; i++)
           newNode=newNode.next;
       return newNode;
   }
   //set data into array
   public void set(AnyType x){set(x,head);}
   private void set(AnyType x,ArrayNode<AnyType> node){
       ArrayNode<AnyType> newNode=node;
       while (newNode.next!=null)
           newNode=newNode.next;
       theSize++;
       newNode.next=new ArrayNode<>(x);
   }
    //remove
   public void remove(AnyType x){remove(x,head);}
   private void remove(AnyType x,ArrayNode<AnyType> node){
       if(!contains(x))
           return;
       while (node.next!=null){
           node=node.next;
           if (node.next.data==x)
               break;
       }
       ArrayNode<AnyType> oldNode=node.next;
       node.next=null;
       node.next=oldNode.next;
   }
}

哈希表實現(xiàn)

public class MyHashTable<AnyType>{
    //define the things that we need to use
    private static final int DEFAULT_SIZE = 10;
    private MyArray<AnyType>[] arrays;
    private int currentSize;

因為我實現(xiàn)的是學(xué)號的存儲
也就是帶0開頭的數(shù)據(jù) 所以我用字符串
這里這個myHash就是我實現(xiàn)的簡單哈希函數(shù),即獲得的數(shù)據(jù)字符串化,得到最后兩個字符

    private int myHash(AnyType x){
        String s=x.toString();
        return Integer.parseInt(s.substring(s.length()-2,s.length()));
    }

初始化哈希表,設(shè)置的默認大小為10,然后進行素數(shù)判斷,如果它不是素數(shù),那么就找到他的下一個素數(shù)作為表的大小。

    //init we should ensure that the table size is prime
    public MyHashTable(){
        ensureTable(nextPrime(DEFAULT_SIZE));
        makeEmpty();
    }
    private void ensureTable(int x){
        arrays=new MyArray[x];
        for (int i = 0; i < arrays.length; i++)
            arrays[i]=new MyArray<>();
    }
    //make the array empty
    public void makeEmpty(){
        currentSize=0;
        for(MyArray<AnyType> myArray:arrays)
            myArray.clear();
    }
    //size and empty
    public int size(){return currentSize;}
    public boolean isEmpty(){return currentSize==0;}

基本方法的實現(xiàn),插入,獲得,刪除,包含

    //contain x
    public boolean contains(AnyType x){
        int position=myHash(x);
        return arrays[position].contains(x);
    }
    //insert x
    public void insert(AnyType x){
        int position=myHash(x);
        if(arrays[position].contains(x))
            return;
        else if(arrays[position].size()==0)
                if(++currentSize>arrays.length)
                    makeBigger();
        arrays[position].set(x);

    }
    //get idx data
    public MyArray<AnyType> get(int idx){ return arrays[idx];}

在這里,如果插入的時候啦,實際的currentSize大于二分之一表的大小了
則進行擴充表
一般擴充表的話,我們是直接兩倍兩倍擴充的。

    //makeBigger
    public void makeBigger() {
        MyArray[] oldArray = arrays;
        arrays = new MyArray[2 * arrays.length];
        for (int i = 0; i < oldArray.length; i++)
            arrays[i] = oldArray[i];
    }

下一個素數(shù)查找,如果他是偶數(shù),則給他加1這樣可以大大減少開銷。
然后進行下一個素數(shù)判斷,奇數(shù)當(dāng)中找素數(shù)。

    //nextPrime
    private int nextPrime(int i){
        if(i%2==0)
            i++;
        for (; !isPrime(i); i+=2);//ensure i is jishu
        return i;
    }

是否為素數(shù)判斷,如果為2則范圍true
如果是1或者為偶數(shù)則返回false
都不滿足則從三開始,他的平方小于輸入的數(shù),用奇數(shù)進行操作,因為用偶數(shù)的話,前面那個2就直接判斷了,所以我們用奇數(shù),大大減少開銷。
我們也可以設(shè)置他的判斷條件是小于輸入的二分之一,但是我們用平方進行判斷大大減少了開銷,而且對于奇數(shù)來說是十分有效果的。

    //is Prime
    private boolean isPrime(int i){
        if(i==2||i==3)
            return true;
        if(i==1||i%2==0)
            return false;
        for (int j = 3; j*j<=i ; j+=2)
            if (i%j==0)
                return false;
        return true;
    }
}

測試

public class test {
    public static void main(String[] args) {
        MyHashTable<String> a=new MyHashTable<>();
        a.insert("001");
        a.insert("01");
        for(int i=1;i<a.get(1).size()+1;i++){
            System.out.println(a.get(1).get(i));
        }
    }
}

結(jié)果

在這里插入圖片描述

到此這篇關(guān)于Java數(shù)據(jù)結(jié)構(gòu)之實現(xiàn)哈希表的分離鏈接法的文章就介紹到這了,更多相關(guān)Java哈希表的分離鏈接法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • java nio基礎(chǔ)使用示例

    java nio基礎(chǔ)使用示例

    傳統(tǒng)的io技術(shù)為阻塞的,java新nio是非阻塞的,注冊一個op_read事件,注冊到selector對象上,當(dāng)有數(shù)據(jù)到來時候,selector回通知之前注冊事件的對象,進行read處理,看面我看看它是如何使用的
    2013-11-11
  • arthas排查jvm中CPU占用過高問題解決

    arthas排查jvm中CPU占用過高問題解決

    這篇文章主要介紹了arthas排查jvm中CPU占用過高問題解決,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-09-09
  • 淺談Java隨機數(shù)的原理、偽隨機和優(yōu)化

    淺談Java隨機數(shù)的原理、偽隨機和優(yōu)化

    這篇文章主要介紹了淺談Java隨機數(shù)的原理、偽隨機和優(yōu)化,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2019-01-01
  • Spring Boot異常處理靜止trace

    Spring Boot異常處理靜止trace

    這篇文章主要介紹了Spring Boot異常處理靜止trace,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2019-12-12
  • Spring Boot 日志配置方法(超詳細)

    Spring Boot 日志配置方法(超詳細)

    默認情況下,Spring Boot會用Logback來記錄日志,并用INFO級別輸出到控制臺。下面通過本文給大家介紹Spring Boot 日志配置方法詳解,感興趣的朋友參考下吧
    2017-07-07
  • Java數(shù)據(jù)結(jié)構(gòu)及算法實例:漢諾塔問題 Hanoi

    Java數(shù)據(jù)結(jié)構(gòu)及算法實例:漢諾塔問題 Hanoi

    這篇文章主要介紹了Java數(shù)據(jù)結(jié)構(gòu)及算法實例:漢諾塔問題 Hanoi,本文直接給出實現(xiàn)代碼,代碼中包含大量注釋,需要的朋友可以參考下
    2015-06-06
  • Java版水果管理系統(tǒng)源碼

    Java版水果管理系統(tǒng)源碼

    這篇文章主要為大家詳細介紹了Java版水果管理系統(tǒng)源碼,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-01-01
  • Java使用JavaMail發(fā)送郵件的方法

    Java使用JavaMail發(fā)送郵件的方法

    這篇文章主要介紹了Java使用JavaMail發(fā)送郵件的方法,結(jié)合實例形式分析了Java使用JavaMail實現(xiàn)郵件發(fā)送的具體步驟與相關(guān)實現(xiàn)代碼,需要的朋友可以參考下
    2016-04-04
  • JavaCV獲取視頻文件時長的方法

    JavaCV獲取視頻文件時長的方法

    這篇文章主要為大家詳細介紹了JavaCV獲取視頻文件時長的方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-07-07
  • Spring Security添加二次認證的項目實踐

    Spring Security添加二次認證的項目實踐

    在用戶自動登錄后,可以通過對密碼進行二次校驗進而確保用戶的真實性,本文就來介紹一下Spring Security添加二次認證的項目實踐,具有一定的參考價值,感興趣的可以了解一下
    2023-12-12

最新評論