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

go數(shù)據(jù)結(jié)構(gòu)和算法BitMap原理及實(shí)現(xiàn)示例

 更新時間:2022年07月22日 14:26:27   作者:haming  
這篇文章主要為大家介紹了go數(shù)據(jù)結(jié)構(gòu)和算法BitMap原理及實(shí)現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

1. BitMap介紹

BitMap可以理解為通過一個bit數(shù)組來存儲特定數(shù)據(jù)的一種數(shù)據(jù)結(jié)構(gòu)。BitMap常用于對大量整形數(shù)據(jù)做去重和查詢。
在這類查找中,我們可以通過map數(shù)據(jù)結(jié)構(gòu)進(jìn)行查找。但如果數(shù)據(jù)量比較大map數(shù)據(jù)結(jié)構(gòu)將會大量占用內(nèi)存。
BitMap用一個比特位來映射某個元素的狀態(tài),所以這種數(shù)據(jù)結(jié)構(gòu)是非常節(jié)省存儲空間的。

BitMap用途

  • BitMap用于數(shù)據(jù)去重
    BitMap可用于數(shù)據(jù)的快速查找,判重。
  • BitMap用于快速排序
    BitMap由于其本身的有序性和唯一性,可以實(shí)現(xiàn)快速排序:將其加入bitmap中,然后再遍歷獲取出來,從而得到排序的結(jié)果。

如何判斷數(shù)字在bit數(shù)組的位置

在后面的代碼中,我們使用[]byte來存儲bit數(shù)據(jù),由于一個byte有8個二進(jìn)制位。因此:

  • 數(shù)字/8=數(shù)字在字節(jié)數(shù)組中的位置。
  • 數(shù)字%8=數(shù)字在當(dāng)前字節(jié)中的位置。
    例如:數(shù)字10,
  • 10/8=1,即數(shù)字10對應(yīng)的字節(jié)數(shù)組的位置為:1
  • 10%8=2,即數(shù)字10對應(yīng)的當(dāng)前字節(jié)的位置為:2

設(shè)置數(shù)據(jù)到bit數(shù)組

  • num/8得到數(shù)字在字節(jié)數(shù)組中的位置 => row
  • num%8得到數(shù)字在當(dāng)前字節(jié)中的位置 => col
  • 將1左移col位,然后和以前的數(shù)據(jù)做|運(yùn)算,這樣就可以將col位置的bit替換成1了。

從bit數(shù)組中清除數(shù)據(jù)

  • num/8得到數(shù)字在字節(jié)數(shù)組中的位置 => row
  • num%8得到數(shù)字在當(dāng)前字節(jié)中的位置 => col
  • 將1左移col位,然后對取反,再與當(dāng)前值做&,這樣就可以將col位置的bit替換成0了。

數(shù)字是否在bit數(shù)組中

  • num/8得到數(shù)字在字節(jié)數(shù)組中的位置 => row
  • num%8得到數(shù)字在當(dāng)前字節(jié)中的位置 => col
  • 將1左移col位,然后和以前的數(shù)據(jù)做&運(yùn)算,若該字節(jié)的值!=0,則說明該位置是1,則數(shù)據(jù)在bit數(shù)組中,否則數(shù)據(jù)不在bit數(shù)組中。

2. Go語言位運(yùn)算

在Go語言中支持以下幾種操作位的方式:

  • & 按位與:兩者全為1結(jié)果為1,否則結(jié)果為0
  • | 按位或:兩者有一個為1結(jié)果為1,否則結(jié)果為0
  • ^ 按位異或:兩者不同結(jié)果為1,否則結(jié)果為0
  • &^ 按位與非:是"與"和"非"操作符的簡寫形式
  • << 按位左移:
  • >> 按位右移:

左移

將二進(jìn)制向左移動,右邊空出的位用0填補(bǔ),高位左移溢出則舍棄該高位。
由于每次移位數(shù)值會翻倍,所以通常用代替乘2操作。當(dāng)然這是建立在移位沒有溢出的情況。
例如:1<<3 相當(dāng)于1×8=8,3<<4 相當(dāng)于3×16=48

右移

將整數(shù)二進(jìn)制向右移動,左邊空出的位用0或者1填補(bǔ)。正數(shù)用0填補(bǔ),負(fù)數(shù)用1填補(bǔ)。
負(fù)數(shù)在內(nèi)存中的二進(jìn)制最高位為符號位——使用1表示,所以為了保證移位之后符號位的正確性,所以需要在高位補(bǔ)1。
相對于左移來說,右移通常用來代替除2操作。
例如:24>>3 相當(dāng)于24÷8=3

使用&^和位移運(yùn)算來給某一位置0

這個操作符通常用于清空對應(yīng)的標(biāo)志位,例如 a = 0011 1010,如果想清空第二位,則可以這樣操作:
a &^ 0000 0010 = 0011 1000

3. BitMap的Go語言實(shí)現(xiàn)

接下來我們給出BitMap的Go語言實(shí)現(xiàn),目前代碼已經(jīng)上傳到github中,下載地址

定義

首先給出BitMap結(jié)構(gòu)的定義:

type BitMap struct {
    bits []byte
    vmax uint
}

創(chuàng)建BitMap結(jié)構(gòu)

func NewBitMap(max_val ...uint) *BitMap {
    var max uint = 8192
    if len(max_val) > 0 && max_val[0] > 0 {
        max = max_val[0]
    }
    bm := &BitMap{}
    bm.vmax = max
    sz := (max + 7) / 8
    bm.bits = make([]byte, sz, sz)
    return bm
}

將數(shù)據(jù)添加到BitMap

func (bm *BitMap)Set(num uint) {
    if num > bm.vmax {
        bm.vmax += 1024
        if bm.vmax < num {
            bm.vmax = num
        }
        dd := int(num+7)/8 - len(bm.bits)
        if dd > 0 {
            tmp_arr := make([]byte, dd, dd)
            bm.bits = append(bm.bits, tmp_arr...)
        }
    }
    //將1左移num%8后,然后和以前的數(shù)據(jù)做|,這樣就替換成1了
    bm.bits[num/8] |= 1 << (num%8)
}

從BitMap中刪除數(shù)據(jù)

func (bm *BitMap)UnSet(num uint) {
    if num > bm.vmax {
        return
    }
    //&^:將1左移num%8后,然后進(jìn)行與非運(yùn)算,將運(yùn)算符左邊數(shù)據(jù)相異的位保留,相同位清零
    bm.bits[num/8] &^= 1 << (num%8)
}

判斷BitMap中是否存在指定的數(shù)據(jù)

func (bm *BitMap)Check(num uint) bool {
    if num > bm.vmax {
        return false
    }
    //&:與運(yùn)算符,兩個都是1,結(jié)果為1
    return bm.bits[num/8] & (1 << (num%8)) != 0
}

以上就是go數(shù)據(jù)結(jié)構(gòu)和算法BitMap原理及實(shí)現(xiàn)示例的詳細(xì)內(nèi)容,更多關(guān)于go數(shù)據(jù)結(jié)構(gòu)算法BitMap的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Go語言基礎(chǔ)學(xué)習(xí)之指針詳解

    Go語言基礎(chǔ)學(xué)習(xí)之指針詳解

    Go 語言中指針是很容易學(xué)習(xí)的,Go 語言中使用指針可以更簡單的執(zhí)行一些任務(wù)。所以本文就來和大家聊聊Go語言中指針的定義與使用,需要的可以參考一下
    2022-12-12
  • 使用Go語言實(shí)現(xiàn)敏感詞過濾功能

    使用Go語言實(shí)現(xiàn)敏感詞過濾功能

    敏感詞過濾,算是一個比較常見的功能,尤其是在內(nèi)容、社交類應(yīng)用中更是如此,本文介紹如何使用Go語言實(shí)現(xiàn)簡單的敏感詞過濾功能,文中通過代碼示例介紹的非常詳細(xì),需要的朋友可以參考下
    2023-12-12
  • Golang實(shí)現(xiàn)優(yōu)雅的將struct轉(zhuǎn)換為map

    Golang實(shí)現(xiàn)優(yōu)雅的將struct轉(zhuǎn)換為map

    在項(xiàng)目實(shí)踐中,有時候我們需要將struct結(jié)構(gòu)體轉(zhuǎn)為map映射表,然后基于map做數(shù)據(jù)裁剪或操作。那么下面我來介紹下常用的兩種轉(zhuǎn)換方式,希望對大家有所幫助
    2023-01-01
  • Go實(shí)現(xiàn)用戶每日限額的方法(例一天只能領(lǐng)三次福利)

    Go實(shí)現(xiàn)用戶每日限額的方法(例一天只能領(lǐng)三次福利)

    這篇文章主要介紹了Go實(shí)現(xiàn)用戶每日限額的方法(例一天只能領(lǐng)三次福利)
    2022-01-01
  • go 協(xié)程返回值處理操作

    go 協(xié)程返回值處理操作

    這篇文章主要介紹了go 協(xié)程返回值處理操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • go-micro微服務(wù)JWT跨域認(rèn)證問題

    go-micro微服務(wù)JWT跨域認(rèn)證問題

    JWT 以 JSON 對象的形式安全傳遞信息。因?yàn)榇嬖跀?shù)字簽名,因此所傳遞的信息是安全的,這篇文章主要介紹了go-micro微服務(wù)JWT跨域認(rèn)證,需要的朋友可以參考下
    2023-01-01
  • Go代碼格式化gofmt的使用方法實(shí)例

    Go代碼格式化gofmt的使用方法實(shí)例

    Golang制定了統(tǒng)一的官方代碼風(fēng)格,并推出gofmt工具(go fmt)來幫助開發(fā)人員格式化代碼到統(tǒng)一的風(fēng)格,下面這篇文章主要給大家介紹了關(guān)于Go代碼格式化gofmt的使用方法,需要的朋友可以參考下
    2023-04-04
  • 一文搞懂Go語言操作Redis的方法

    一文搞懂Go語言操作Redis的方法

    Redis是一個開源的內(nèi)存數(shù)據(jù)庫,在項(xiàng)目開發(fā)中redis的使用也比較頻繁,本文介紹了Go語言中g(shù)o-redis庫的基本使用。感興趣的小伙伴們可以參考借鑒一下
    2022-09-09
  • Go?數(shù)據(jù)結(jié)構(gòu)之二叉樹詳情

    Go?數(shù)據(jù)結(jié)構(gòu)之二叉樹詳情

    這篇文章主要介紹了?Go?數(shù)據(jù)結(jié)構(gòu)之二叉樹詳情,二叉樹是一種數(shù)據(jù)結(jié)構(gòu),在每個節(jié)點(diǎn)下面最多存在兩個其他節(jié)點(diǎn)。即一個節(jié)點(diǎn)要么連接至一個、兩個節(jié)點(diǎn)或不連接其他節(jié)點(diǎn),下文基于GO語言展開二叉樹結(jié)構(gòu)詳情,需要的朋友可以參考一下
    2022-05-05
  • Go語言實(shí)現(xiàn)UDP版聊天小工具的示例詳解

    Go語言實(shí)現(xiàn)UDP版聊天小工具的示例詳解

    這篇文章主要為大家詳細(xì)介紹了如何利用Go語言實(shí)現(xiàn)聊天小工具(UDP版),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-03-03

最新評論