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

Java必備知識之位運算及常見進制解讀

 更新時間:2021年10月09日 17:02:53   作者:吾日三省賈斯汀  
從現(xiàn)代計算機中所有的數(shù)據(jù)二進制的形式存儲在設(shè)備中。即 0、1 兩種狀態(tài),計算機對二進制數(shù)據(jù)進行的運算(+、-、*、/)都是叫位運算,即將符號位共同參與運算的運算

您好,我是賈斯汀,歡迎又進來學(xué)習(xí)啦!

在這里插入圖片描述

【學(xué)習(xí)背景】

學(xué)習(xí)Java的小伙伴,都知道想要提升個人技術(shù)水平,閱讀JDK源碼少不了,但是說實話還是有些難度的,底層源碼實現(xiàn)的原理離不開各種常用的數(shù)據(jù)結(jié)構(gòu)和算法,很多時候還會用到各種位運算,比如面試必問和工作寫爛透了的HashMap,就一個put(key,value)添加元素的底層實現(xiàn),就用到了各種位運算知識,不對位運算略知一二,你還真讀不懂它的源碼,所以本文主要對Java中的幾種位運算以及常見進制的說明,還會以HashMap底層實現(xiàn)添加元素四部曲展開說明,希望能提高提升自己對源碼的理解,也希望能幫助到有需要的小伙伴~

進入正文~

常見幾種進制?

在這里插入圖片描述

  • 二進制(Binary)

數(shù)值范圍0,1,滿2進1
以0b或0B開頭
bit比特是計算機最小存儲單元,1個bit占用1個二進制位即0或1
1個byte字節(jié)有8個bit即占用8個二進制位
int整型4字節(jié)占用32個二進制位
二進制左半部分表示高位,右半部分為低位
二進制最高位為0表示正數(shù),最高位為1表示負數(shù)
二進制原碼取反得到反碼,反碼補1得到補碼,負數(shù)使用補碼表示

  • 八進制(Octal)

數(shù)值范圍0-7,滿8進1
以數(shù)字0開頭表示

  • 十進制(Decimal)

數(shù)值范圍0-9 ,滿10進1
日常阿拉伯?dāng)?shù)字即十進制

  • 十六進制(Hexadecimal)

數(shù)值范圍0-9及A-F,滿16進1
以0x或0X開頭表示。 此處的A-F不區(qū)分大小寫

Java八種按位運算?

在這里插入圖片描述

  • 按位與(&)

都為1則得1

  • 按位或(|)

有一個為1即得1

  • 按位異或(^)

不同得1,相同得0

  • 按位取反(~)

取反即1變0、0變1

  • 按位左移(<<)

按位左移幾位,高位會被截掉幾位,正負數(shù),低位都會被補幾個0

  • 按位右移(>>)

按位右移幾位,低位就會被截掉幾位,正數(shù)高位會被補幾個0,負數(shù)高位會被補幾個1

  • 按位無符號右移(>>>)

按位右移幾位,低位就會被截掉幾位,正負數(shù)數(shù)高位會被補幾個0

  • 按位無符號左移(<<<)

按位左移幾位,高位就會被截掉幾位,正負數(shù)數(shù)低位都會被補幾個0

HashMap添加元素四步曲用到的位運算?

前奏:HashMap如何添加一個元素?

HashMap底層數(shù)據(jù)結(jié)構(gòu)是數(shù)組+鏈表,通過put(K key, V value)方法添加元素,底層四步曲如下:

  • 第一步曲:根據(jù)key得到hashCode值
  • 第二步曲:根據(jù)hashCode值計算出hash值
  • 第三步曲:根據(jù)hash值計算出元素(key/value)最終要放在哪個數(shù)組index下標(biāo)
  • 第四步曲:最后根據(jù)元素(key/value)新建節(jié)點并保存到指定數(shù)組index下標(biāo)位置

Java HashMap添加元素的示例代碼:

        HashMap<Object, Object> map = new HashMap<>();
        map.put("name","Justin");

HashMap底層put(key,value)方法源碼:

    public V put(K key, V value) {
        return putVal(hash(key), key, value, false, true);
    }

接下來將解讀底層源碼用到哪些位運算,有什么奧妙之處

第一步曲

根據(jù)key得到hashCode值
可以看到hash值計算的過程就用到了^(異或)和>>>(無符號右移)兩種位運算

    static final int hash(Object key) {
        int h;
        return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
    }

這里key是字符串"name",String重寫了計算字符串hashCode值的hashCode()方法,源碼如下:

在這里插入圖片描述


計算得到hashCode值為3373707

第二步曲

根據(jù)hashCode值計算出hash值
(h = key.hashCode()) ^ (h >>> 16)(3373707) ^ (3373707 >>> 16)
3373707二進制表達
0000000001100110111101010001011
h >>> 16二進制表達
00000000000000000000000000110011
根據(jù)^異或運算原理即不同得1,相同得0得到3373707 ^ (3373707 >>>16)二進制結(jié)果為:
0000000001100110111101010111000
進制在線轉(zhuǎn)換:http://tools.jb51.net/transcoding/hexconvert

在這里插入圖片描述

即計算key的hash值得到3373752,斷點往后查看hash值剛好也是這個值

第三步曲

根據(jù)hash值計算出元素(key/value)最終要放在哪個數(shù)組index下標(biāo)
公式:i = (n - 1) & hash
這里就用到了&按位與運算(都為1則得1)

在這里插入圖片描述

公式(n - 1) & hash 的奧妙之處在于,n表示HashMap中的數(shù)組容量大小,并且剛好是16,32,64…2的次方,這種情況其實是等效于 hash % n 取模,計算出的數(shù)組index下標(biāo)值一樣,還能夠保證不會數(shù)組下標(biāo)越界

但是HashMap這里沒有使用%取模,因為hash值是int整型即十進制數(shù)值,使用%取模會先將內(nèi)存數(shù)據(jù)轉(zhuǎn)成十進制再進行運算,多了這部分的性能開銷,因此效率比較低

HashTable底層倒是用的%取模,hash值與十六進制0x7FFFFFFF做按位與運算目的是為了保證hash值始終是正數(shù)

在這里插入圖片描述

有的小伙伴可能會問了,使用%取模計算,那這里為啥HashTable還在用,我想說的是其實也可以優(yōu)化,只不過HashTable本身就是主打synchronized線程安全,也就不考慮優(yōu)化%取模為位運算了

在這里插入圖片描述

第四步曲

最后根據(jù)元素(key/value)新建節(jié)點并保存到指定數(shù)組index下標(biāo)位置

    Node<K,V> newNode(int hash, K key, V value, Node<K,V> next) {
        return new Node<>(hash, key, value, next);
    }

終曲:為什么HashMap底層源碼用這么多位運算?

關(guān)于位運算的使用,文中在介紹第三步曲時,也提到了HashMap計算數(shù)組下標(biāo)使用%取模和位運算的問題,使用于位運算的奧妙之處在直接從內(nèi)存讀取數(shù)據(jù)進行計算,不需要轉(zhuǎn)成十進制,如果使用%取模需要先轉(zhuǎn)成十進制,有性能開銷,效率比較低

HashMap底層除了文中提到的^按位異或、>>>無符號右移、&按位與位運算,其實在HashMap的擴容機制resize()中,還用到了<<左移運算
oldCap << 1

在這里插入圖片描述


這里oldCap << 1剛好是兩倍,可以總結(jié)的說一個數(shù)與n進行左移運算,結(jié)果為這個數(shù)乘以2的n次方
oldCap << 1 等值 oldCap = oldCap * (2的n次方)
同理,一個數(shù)與n進行右移運算結(jié)果為這個數(shù)除以2的n次方
oldCap >> 1 等值 oldCap = oldCap / (2的n次方)

**

到此這篇關(guān)于Java必備知識之位運算及常見進制解讀的文章就介紹到這了,更多相關(guān)Java 位運算內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 實例講解Java的MyBatis框架對MySQL中數(shù)據(jù)的關(guān)聯(lián)查詢

    實例講解Java的MyBatis框架對MySQL中數(shù)據(jù)的關(guān)聯(lián)查詢

    這里我們來以實例講解Java的MyBatis框架對MySQL中數(shù)據(jù)的關(guān)聯(lián)查詢,包括一對多、多對一的關(guān)聯(lián)查詢以及自身關(guān)聯(lián)映射的方法等,需要的朋友可以參考下
    2016-06-06
  • Java vector的詳解及實例

    Java vector的詳解及實例

    這篇文章主要介紹了Java vector的詳解及實例的相關(guān)資料,需要的朋友可以參考下
    2017-03-03
  • Fluent Mybatis 批量更新的使用

    Fluent Mybatis 批量更新的使用

    本文主要介紹了Fluent Mybatis 批量更新的使用,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • Spring data jpa的使用與詳解(復(fù)雜動態(tài)查詢及分頁,排序)

    Spring data jpa的使用與詳解(復(fù)雜動態(tài)查詢及分頁,排序)

    這篇文章主要介紹了Spring data jpa的使用與詳解(復(fù)雜動態(tài)查詢及分頁,排序),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-11-11
  • Java編程刪除鏈表中重復(fù)的節(jié)點問題解決思路及源碼分享

    Java編程刪除鏈表中重復(fù)的節(jié)點問題解決思路及源碼分享

    這篇文章主要介紹了Java編程刪除鏈表中重復(fù)的節(jié)點問題解決思路及源碼分享,具有一定參考價值,這里分享給大家,供需要的朋友了解。
    2017-10-10
  • Java高并發(fā)下請求合并處理方式

    Java高并發(fā)下請求合并處理方式

    這篇文章主要介紹了Java高并發(fā)下請求合并處理方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-08-08
  • Java基礎(chǔ)之詳細總結(jié)五種常用運算符

    Java基礎(chǔ)之詳細總結(jié)五種常用運算符

    在通常代碼邏輯處理中,我們常常都會使用到運算符,今天我們就詳細了解一下運算符的使用以及分類.運算符是對常量或者變量進行操作的符號,它分為算術(shù)運算符,賦值運算符,比較運算符,邏輯運算符以及位運算符.需要的朋友可以參考下
    2021-05-05
  • java多線程從入門到精通看這篇就夠了

    java多線程從入門到精通看這篇就夠了

    熟悉 Java 多線程編程的同學(xué)都知道,當(dāng)我們線程創(chuàng)建過多時,容易引發(fā)內(nèi)存溢出,因此我們就有必要使用線程池的技術(shù)了,今天通過本文給大家分享java多線程從入門到精通的相關(guān)知識,一起看看吧
    2021-06-06
  • springcloud中RabbitMQ死信隊列與延遲交換機實現(xiàn)方法

    springcloud中RabbitMQ死信隊列與延遲交換機實現(xiàn)方法

    死信隊列是消息隊列中非常重要的概念,同時我們需要業(yè)務(wù)場景中都需要延遲發(fā)送的概念,比如12306中的30分鐘后未支付訂單取消,那么本期,我們就來講解死信隊列,以及如何通過延遲交換機來實現(xiàn)延遲發(fā)送的需求,感興趣的朋友一起看看吧
    2022-05-05
  • 一文帶你了解Java選擇排序的原理與實現(xiàn)

    一文帶你了解Java選擇排序的原理與實現(xiàn)

    選擇排序:(Selection sort)是一種簡單直觀的排序算法,也是一種不穩(wěn)定的排序方法。本文主要為大家介紹一下選擇排序的原理與實現(xiàn),希望對大家有所幫助
    2022-11-11

最新評論