0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

如何使用位操作符實(shí)現(xiàn)低級別的位掩碼

科技綠洲 ? 來源:Java技術(shù)指北 ? 作者:Java技術(shù)指北 ? 2023-10-07 16:52 ? 次閱讀

在本文中,我們來看看如何使用位操作符實(shí)現(xiàn)低級別的位掩碼。我們將看到我們?nèi)绾螌⒁粋€(gè)單一的int變量作為一個(gè)單獨(dú)的數(shù)據(jù)容器。

位掩碼

位掩碼允許我們在一個(gè)數(shù)字變量中存儲多個(gè)值。我們不再把這個(gè)變量看作一個(gè)整數(shù),而是把它的每一個(gè)比特當(dāng)作一個(gè)獨(dú)立的值。

因?yàn)橐粋€(gè)比特可以等于 0 或 1,我們也可以把它看成是 false 或 true 。我們也可以把一組比特切開,把它們當(dāng)作一個(gè)較小的數(shù)字變量甚至是一個(gè)String。

舉個(gè)例子

假設(shè)我們有一個(gè)最小的內(nèi)存空間,并且需要在一個(gè)int變量中存儲所有關(guān)于用戶賬戶的信息。前八位(來自32個(gè)可用位)將存儲boolean信息,如 "該賬戶是否激活?"或 "該賬戶是否溢價(jià)?"

至于剩下的24位,我們將把它們轉(zhuǎn)換成三個(gè)字符,作為用戶的標(biāo)識符。

編碼

我們的用戶將有一個(gè)標(biāo)識符 "AAA",他將有一個(gè)活躍的高級賬戶(存儲在前兩個(gè)比特)。在二進(jìn)制表示中,它將看起來像。

String stringRepresentation = "01000001010000010100000100000011";

使用內(nèi)置的Integer#parseUnsignedInt方法,可以很容易地將其編碼為一個(gè)int變量。

int intRepresentation = Integer.parseUnsignedInt(stringRepresentation, 2);
assertEquals(intRepresentation, 1094795523);

解碼

這個(gè)過程也可以用 Integer#toBinaryString 方法來反轉(zhuǎn)。

String binaryString = Integer.toBinaryString(intRepresentation);
String stringRepresentation = padWithZeros(binaryString);
assertEquals(stringRepresentation, "01000001010000010100000100000011");

提取一個(gè)比特

第一比特

如果我們想檢查我們賬戶變量的第一位,我們只需要使用順位 and 運(yùn)算符和數(shù)字 1 作為掩碼。因?yàn)閿?shù)字 1在二進(jìn)制形式中只有第一位被設(shè)置為1,其余的都是0,它將從我們的變量中刪除所有的位,只留下第一個(gè)完整的位。

10000010100000101000001000000011
00000000000000000000000000000001
-------------------------------- &
00000000000000000000000000000001

然后我們需要檢查產(chǎn)生的值是否不等于零。

intRepresentation & 1 != 0

任意位置的位

如果我們想檢查其他的位,我們需要?jiǎng)?chuàng)建一個(gè)適當(dāng)?shù)难诖a,這個(gè)掩碼需要在給定的位置上有一個(gè)位設(shè)置為1,其余的設(shè)置為0。最簡單的方法是對我們已有的掩碼進(jìn)行移位。

1 < < (position - 1)

上面這行代碼的位置變量設(shè)置為3,將把我們的掩碼從 00000000000000000000000000000001 變成

00000000000000000000000000000100

因此,現(xiàn)在,比特方程將看起來像這樣。

10000010100000101000001000000011
00000000000000000000000000000100
-------------------------------- &
00000000000000000000000000000000

把所有這些放在一起,我們可以寫一個(gè)方法來提取給定位置上的單個(gè)比特。

private boolean extractValueAtPosition(int intRepresentation, int position) {
    return ((intRepresentation) & (1 < < (position - 1))) != 0;
}

為了達(dá)到同樣的效果,我們也可以將intRepresentation變量向相反方向移動(dòng),而不是改變掩碼。

提取多個(gè)比特

我們可以用類似的方法從一個(gè)整數(shù)中提取多個(gè)比特。讓我們提取我們的用戶帳戶變量的最后三個(gè)字節(jié),并將其轉(zhuǎn)換為一個(gè)字符串。首先,我們需要通過將變量向右移動(dòng)來擺脫前八位的影響。

int lastThreeBites = intRepresentation > > 8;
String stringRepresentation = getStringRepresentation(lastThreeBites);
assertEquals(stringRepresentation, "00000000010000010100000101000001");

我們?nèi)匀挥?2位,因?yàn)?code>int總是有32位。然而,現(xiàn)在我們只對前24位感興趣,其余的都是零,會很容易被忽略。我們創(chuàng)建的int變量可以很容易地用作整數(shù)ID,但是因?yàn)槲覀兿胗幸粋€(gè)字符串ID,所以我們還有一個(gè)步驟要做。

我們將把二進(jìn)制的字符串表示法分成8個(gè)字符的組,把它們解析成char變量,然后把它們連接成一個(gè)最終的String。

為了方便起見,我們還將忽略空字節(jié)。

Arrays.stream(stringRepresentation.split("(?<=G.{8})"))
  .filter(eightBits - > !eightBits.equals("00000000"))
  .map(eightBits - > (char)Integer.parseInt(eightBits, 2))
  .collect(StringBuilder::new, StringBuilder::append, StringBuilder::append)
  .toString();

應(yīng)用一個(gè)比特掩碼

我們也可以創(chuàng)建一個(gè)掩碼來同時(shí)檢查許多位,而不是提取和檢查單個(gè)位的值。我們想檢查我們的用戶是否有一個(gè)活躍的高級賬戶,所以他的變量的前兩個(gè)比特都設(shè)置為1。

我們可以用以前的方法分別檢查它們,但創(chuàng)建一個(gè)掩碼來選擇它們,會更快。

int user = Integer.parseUnsignedInt("00000000010000010100000101000001", 2);
int mask = Integer.parseUnsignedInt("00000000000000000000000000000011", 2);
int masked = user & mask;

因?yàn)槲覀兊挠脩粲幸粋€(gè)活躍的賬戶,但它不是高級賬戶,所以被屏蔽的值將只有第一個(gè)比特被設(shè)置為1。

assertEquals(getStringRepresentation(masked), "00000000000000000000000000000001");

現(xiàn)在,我們可以輕松而廉價(jià)地?cái)嘌砸粋€(gè)用戶是否符合我們的條件。

assertFalse((user & mask) == mask);

總結(jié)

在本教程中,我們學(xué)習(xí)了如何使用位運(yùn)算符來創(chuàng)建位掩碼,并應(yīng)用它們來從整數(shù)中提取二進(jìn)制信息。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報(bào)投訴
  • 數(shù)據(jù)
    +關(guān)注

    關(guān)注

    8

    文章

    6715

    瀏覽量

    88308
  • 二進(jìn)制
    +關(guān)注

    關(guān)注

    2

    文章

    761

    瀏覽量

    41476
  • 內(nèi)存
    +關(guān)注

    關(guān)注

    8

    文章

    2903

    瀏覽量

    73536
  • 運(yùn)算符
    +關(guān)注

    關(guān)注

    0

    文章

    166

    瀏覽量

    11013
收藏 人收藏

    評論

    相關(guān)推薦

    Linux命令中“!”操作符的用法

    Linux中的'!'符號或操作符可以用作邏輯否定運(yùn)算,也可以用于在歷史記錄中獲取命令并進(jìn)行修改或運(yùn)行以前執(zhí)行過的命令。
    發(fā)表于 07-05 10:07 ?1376次閱讀

    MATLAB操作符和特殊字符

    MATLAB操作符和特殊字符* 矩陣乘法 .* 數(shù)組乘法 ^ 矩陣冪 .^ 數(shù)組冪 \ 左除或反斜杠 / 右除或斜杠 ./ 數(shù)組除 Kron Kronecker張量積 .. 父目錄 … 繼續(xù)
    發(fā)表于 09-22 16:05

    C語言,操作符優(yōu)先級順序。

    ;、==、!=) 〉邏輯運(yùn)算(特別要說明,按與、或的優(yōu)先級高于邏輯與、或),接下來就是三目運(yùn)算?=然后是賦值操作符等號,最后是逗號,。運(yùn)算
    發(fā)表于 01-16 17:30

    操作符的相關(guān)資料分享

    嵌入式C語言入門——操作符
    發(fā)表于 12-15 06:50

    操作C語言支持的幾種操作符是哪些呢

    一、操作C語言支持的6種操作符如下:1.不改變其他的值的狀況下,對某幾個(gè)位進(jìn)行設(shè)值。方法:先對需要設(shè)置的
    發(fā)表于 12-22 08:07

    操作符及其常用方式簡要概述

    目錄前言一、操作符及其常用方式二、實(shí)例應(yīng)用解析(嵌入式筆試??迹┣把?b class='flag-5'>位操作在單片機(jī)的C語言開發(fā)中經(jīng)常會用到,該操作主要用于讀寫寄存器,這篇
    發(fā)表于 02-17 06:33

    C語言編程二進(jìn)制操作符

    C語言編程二進(jìn)制操作符
    發(fā)表于 03-30 14:09 ?6次下載

    使用51單片機(jī)內(nèi)核的匯編操作符筆記

    本文檔的主要內(nèi)容詳細(xì)介紹的是使用51單片機(jī)內(nèi)核的匯編操作符筆記。
    發(fā)表于 07-19 17:38 ?1次下載
    使用51單片機(jī)內(nèi)核的匯編<b class='flag-5'>操作符</b>筆記

    C++:詳談取地址操作符重載

    取址操作符重載函數(shù)返回值為該類型的指針,無參數(shù)。
    的頭像 發(fā)表于 06-29 11:59 ?3383次閱讀
    C++:詳談取地址<b class='flag-5'>操作符</b>重載

    C++之操作重載學(xué)習(xí)的總結(jié)(二)

    復(fù)數(shù)的概念可以通過自定義類實(shí)現(xiàn);復(fù)數(shù)中的運(yùn)算操作可以通過操作符重載實(shí)現(xiàn);賦值操作符只能通過成員函數(shù)實(shí)現(xiàn)
    的頭像 發(fā)表于 12-24 16:26 ?518次閱讀

    C++之操作符重載學(xué)習(xí)的總結(jié)

    操作符重載是c++的強(qiáng)大特性之一;操作符重載的本質(zhì)是通過函數(shù)擴(kuò)展操作符的功能;operator 關(guān)鍵字是實(shí)現(xiàn)操作符重載的關(guān)鍵。
    的頭像 發(fā)表于 12-24 16:36 ?594次閱讀

    C語言操作

    目錄前言一、操作符及其常用方式二、實(shí)例應(yīng)用解析(嵌入式筆試常考)前言 操作在單片機(jī)的C語言開發(fā)中經(jīng)常會用到,該操作主要用于讀寫寄存器,
    發(fā)表于 12-22 19:09 ?12次下載
    C語言<b class='flag-5'>位</b><b class='flag-5'>操作</b>

    Verilog基礎(chǔ):幾個(gè)常用的按操作符

    操作符是對二進(jìn)制進(jìn)行操作的運(yùn)算。
    的頭像 發(fā)表于 11-09 10:57 ?1131次閱讀

    Verilog基礎(chǔ):介紹幾個(gè)常用的按操作符

    操作符是對二進(jìn)制進(jìn)行操作的運(yùn)算。以下是一些常用的操作
    的頭像 發(fā)表于 11-09 10:59 ?1288次閱讀

    “+”操作符的使用技巧

    這篇寫個(gè)平時(shí)易被忽略的小知識點(diǎn),一元 + 操作符的使用技巧。
    的頭像 發(fā)表于 12-28 13:27 ?404次閱讀