Bitvis drift i VB.NET

VB.NET støtter ikke operasjoner på bitnivå direkte. Framework 1.1 (VB.NET 2003) introduserte bit shift-operatører (<< og >>), men ingen generell måte å manipulere individuelle biter er tilgjengelig. Bitoperasjoner kan være veldig nyttig. For eksempel kan programmet ditt måtte grensesnitt med et annet system som krever bitmanipulering. Men i tillegg er det mange triks som kan gjøres ved bruk av individuelle biter. Denne artikkelen kartlegger hva som kan gjøres med bitmanipulering ved hjelp av VB.NET.

Du må forstå bitvise operatører før noe annet. I VB.NET er disse:

  • Og
  • Eller
  • xor
  • Ikke

Bitvis betyr ganske enkelt at operasjonene kan utføres på to binære tall bit for bit. Microsoft bruker sannhetstabeller for å dokumentere bitvise operasjoner. Sannhetstabellen for Og er:

1. bit 2. bit Resultat
1 1 1
1 0 0
0 1 0
0 0 0

På skolen min lærte de Karnaugh kart i stedet. Karnaugh-kartet for alle fire operasjoner er vist på illustrasjonen nedenfor.


Klikk her for å vise illustrasjonen
Klikk på Tilbake-knappen i nettleseren for å gå tilbake

instagram viewer

Her er et enkelt eksempel ved å bruke Og operasjon med to, fire bit binære tall:

Resultatet av 1100 Og 1010 er 1000.

Det er fordi 1 Og 1 er 1 (den første biten) og resten er 0.

Til å begynne med, la oss se på bitoperasjonene som er støttet direkte i VB.NET: litt skiftende. Selv om både venstre- og høyreskift er tilgjengelig, fungerer de på samme måte, så bare venstre skift vil bli diskutert. Bitskifting brukes ofte i kryptografi, bildebehandling og kommunikasjon.

VB.NETs bitskiftende operasjoner ...

  • Arbeid bare med de fire heltallstypene: byte, Kort, Integer, og Lang
  • er aritmetikk skiftoperasjoner. Det betyr at biter som er forskjøvet forbi slutten av resultatet blir kastet, og bitposisjonene åpnet i den andre enden er satt til null. Alternativet kalles sirkulær bitforskyvning og bitene forskjøvet forbi den ene enden legges ganske enkelt til den andre. VB.NET støtter ikke sirkulær bitskifting direkte. Hvis du trenger det, må du kode det på gammeldags måte: multiplisere eller dele med 2.
  • Generer aldri et overflow-unntak. VB.NET tar seg av eventuelle problemer, og jeg vil vise deg hva det betyr. Som nevnt kan du kode din egen bitskifting ved å multiplisere eller dele med 2, men hvis du bruker "kode din egen" tilnærming, må du teste for overflødige unntak som kan føre til at programmet ditt gjør det brak.

En standard bitforskyvningsoperasjon vil se slik ut:

Dim StartValue Som heltall = 14913080
Dim ValueAfterShifting Som heltall
ValueAfterShifting = StartValue << 50

Med ord tar denne operasjonen den binære verdien 0000 0000 1110 0011 1000 1110 0011 1000 (14913080 er den tilsvarende desimalverdien - legg merke til at det bare er en serie på 3 0'er og 3 1er gjentatt noen ganger) og forskyver den 50 plasser igjen. Men siden et helt tall bare er 32 bit langt, er det meningsløst å skifte 50 plasser. VB.NET løser dette problemet ved maskerings skiftetellingen med en standardverdi som samsvarer med datatypen som brukes. I dette tilfellet, ValueAfterShifting er en Integer så det maksimale som kan forskyves er 32 biter. Standard maskeverdi som fungerer er 31 desimal eller 11111.

Maske betyr at verdien, i dette tilfellet 50, er Oged med masken. Dette gir maksimalt antall biter som faktisk kan forskyves for den datatypen.

I desimal:

50 og 31 er 18 - Maksimum antall biter som kan forskyves

Det gir faktisk mer mening i binær. De høye ordrebitene som ikke kan brukes til skiftoperasjonen blir ganske enkelt fjernet.

110010 Og 11111 er 10010

Når kodebiten kjøres, er resultatet 954204160 eller, binært, 0011 1000 1110 0000 0000 0000 0000 0000. De 18 bitene på venstre side av det første binære nummeret forskyves og de 14 bitene på høyre side forskyves til venstre.

Det andre store problemet med å skifte biter er hva som skjer når antall steder å skifte er et negativt tall. La oss bruke -50 som antall biter for å skifte og se hva som skjer.

ValueAfterShifting = StartValue << -50

Når dette kodebiten kjøres, får vi -477233152 eller 1110 0011 1000 1110 0000 0000 0000 0000 i binær. Antallet er forskjøvet 14 plasser igjen. Hvorfor 14? VB.NET antar at antall plasser er et usignert heltall og gjør et Og operasjon med samme maske (31 for heltal).

1111 1111 1111 1111 1111 1111 1100 1110
0000 0000 0000 0000 0000 0000 0001 1111
(Og)
0000 0000 0000 0000 0000 0000 0000 1110

1110 i binær er 14 desimaler. Legg merke til at dette er motsatt for å skifte positive 50 plasser.

På neste side går vi videre til noen andre bitoperasjoner, med start Xor-kryptering!

Jeg nevnte at en bruk av bitoperasjoner er kryptering. Xor-kryptering er en populær og enkel måte å "kryptere" en fil på. I artikkelen min, Very Simple Encryption using VB.NET, viser jeg deg en bedre måte å bruke strengmanipulering i stedet. Men Xor-kryptering er så vanlig at den fortjener å i det minste bli forklart.

Å kryptere en tekststreng betyr å oversette den til en annen tekststreng som ikke har et åpenbart forhold til den første. Du trenger også en måte å dekryptere den igjen. Xor-kryptering oversetter den binære ASCII-koden for hvert tegn i strengen til et annet tegn ved å bruke Xor-operasjonen. For å gjøre denne oversettelsen, trenger du et annet nummer for å bruke i Xor. Dette andre tallet kalles nøkkelen.

Xor-kryptering kalles en "symmetrisk algoritme". Dette betyr at vi også kan bruke krypteringsnøkkelen som dekrypteringsnøkkel.

La oss bruke "A" som nøkkel og kryptere ordet "Basic". ASCII-koden for "A" er:

0100 0001 (desimal 65)

ASCII-koden for Basic er:

B - 0100 0010
a - 0110 0001
s - 0111 0011
i - 0110 1001
c - 0110 0011

De xor av hver av disse er:

0000 0011 - desimal 3
0010 0000 - desimal 32
0011 0010 - desimal 50
0010 1000 - desimal 40
0010 0010 - desimal 34

Denne lille rutinen gjør susen:

- Xor-kryptering -
Dim i som kort
ResultString. Tekst = ""
Dim KeyChar som heltall
KeyChar = Asc (EncryptionKey. Tekst)
For i = 1 til Len (InputString. Tekst)
ResultString. Tekst & = _
Chr (KeyChar Xor _
Asc (Midt (InputString) Tekst, i, 1)))
neste

Resultatet kan sees i denne illustrasjonen:


Klikk her for å vise illustrasjonen
Klikk på Tilbake-knappen i nettleseren for å gå tilbake

For å reversere krypteringen, bare kopier og lim inn strengen fra Result TextBox tilbake i String TextBox og klikk på knappen igjen.

Et annet eksempel på noe du kan gjøre med bitvise operatører er å bytte to heltall uten å erklære en tredje variabel for midlertidig lagring. Dette er den slags ting de pleide å gjøre i samlingsspråklige programmer for år siden. Det er ikke for nyttig nå, men du kan vinne en innsats en dag hvis du kan finne noen som ikke tror du kan gjøre det. I alle fall hvis du fortsatt har spørsmål om hvordan xor fungerer, å jobbe gjennom dette skulle få dem til å hvile. Her er koden:

Dim FirstInt Som heltall
Dim SecondInt Som heltall
FirstInt = CInt (FirstIntBox. Tekst)
SecondInt = CInt (SecondIntBox. Tekst)
FirstInt = FirstInt Xor SecondInt
SecondInt = FirstInt Xor SecondInt
FirstInt = FirstInt Xor SecondInt
ResultBox. Tekst = "Første heltall:" & _
FirstInt. ToString & "-" & _
"Andre heltall:" & _
SecondInt. toString

Og her er koden i aksjon:


Klikk her for å vise illustrasjonen
Klikk på Tilbake-knappen i nettleseren for å gå tilbake

Å finne ut nøyaktig hvorfor dette fungerer vil bli stående som "som en øvelse for studenten".

På neste side når vi målet: General Bit Manipulation

Selv om disse triksene er morsomme og lærerike, er de fortsatt ingen erstatning for generell bitmanipulering. Hvis du virkelig kommer ned på nivået av biter, er det du ønsker en måte å undersøke individuelle biter, angi dem eller endre dem. Det er den virkelige koden som mangler fra .NET.

Kanskje grunnen til at det mangler er at det ikke er så vanskelig å skrive underutgaver som oppnår det samme.

En typisk grunn til at du kanskje vil gjøre dette, er å opprettholde det som noen ganger kalles a flaggbyte. Noen applikasjoner, spesielt de som er skrevet på språk på lavt nivå som assembler, vil opprettholde åtte boolske flagg i en enkelt byte. For eksempel inneholder en 6502-prosessorchips statusregister denne informasjonen i en enkelt 8-bitbyte:

Bit 7. Negativt flagg
Bit 6. Overløpsflagg
Bit 5. Ubrukt
Bit 4. Bryt flagg
Bit 3. Desimal flagg
Bit 2. Avbryt-deaktiver flagg
Bit 1. Nullflagg
Bit 0. Bær flagg

(fra Wikipedia)

Hvis koden din må fungere med denne typen data, trenger du generell bruk av bitmanipulasjonskode. Denne koden vil gjøre jobben!

ClearBit Sub tømmer den 1 baserte, niende biten
'(MyBit) av et helt tall (MyByte).
Sub ClearBit (ByRef MyByte, ByVal MyBit)
Dim BitMask As Int16
"Lag en bitmaske med 2 til nth power bit set:
BitMask = 2 ^ (MyBit - 1)
"Tøm den niende biten:
MyByte = MyByte og ikke BitMask
Slutt sub
'ExamineBit-funksjonen vil returnere True eller False
avhengig av verdien av den 1 baserte, niende biten (MyBit)
'av et helt tall (MyByte).
Funksjon ExamineBit (ByVal MyByte, ByVal MyBit) Som boolsk
Dim BitMask As Int16
BitMask = 2 ^ (MyBit - 1)
ExamineBit = ((MyByte And BitMask)> 0)
Sluttfunksjon
'SetBit Sub vil stille inn den 1 baserte, niende biten
'(MyBit) av et helt tall (MyByte).
Sub SetBit (ByRef MyByte, ByVal MyBit)
Dim BitMask As Int16
BitMask = 2 ^ (MyBit - 1)
MyByte = MyByte Eller BitMask
Slutt sub
'ToggleBit Sub vil endre tilstanden
'av den 1 baserte, niende biten (MyBit)
'av et helt tall (MyByte).
Sub ToggleBit (ByRef MyByte, ByVal MyBit)
Dim BitMask As Int16
BitMask = 2 ^ (MyBit - 1)
MyByte = MyByte Xor BitMask
Slutt sub

For å demonstrere koden kaller denne rutinen den (parametere som ikke er kodet på Click Sub):

Private Sub ExBitCode_Click (...
Dim Byte1, Byte2 Som Byte
Dim MyByte, MyBit
Dim StatusOfBit Som boolsk
Dim SelectedRB som streng
StatusLine. Tekst = ""
SelectedRB = GetCheckedRadioButton (Me) .navn
Byte1 = ByteNum. Tekst 'Nummer som skal konverteres til bitflagg
Byte2 = BitNum. Tekst 'Bit som skal kobles til
'Følgende tømmer bydelen med høy ordre og returnerer bare
'lav ordre byte:
MyByte = Byte1 Og & HFF
MyBit = Byte2
Velg Case SelectedRB
Sak "ClearBitButton"
ClearBit (MyByte, MyBit)
StatusLine. Tekst = "Ny byte:" & MyByte
Sak "ExamineBitButton"
StatusOfBit = ExamineBit (MyByte, MyBit)
StatusLine. Tekst = "Bit" & MyBit & _
"er" & StatusOfBit
Sak "SetBitButton"
SetBit (MyByte, MyBit)
StatusLine. Tekst = "Ny byte:" & MyByte
Sak "ToggleBitButton"
ToggleBit (MyByte, MyBit)
StatusLine. Tekst = "Ny byte:" & MyByte
Slutt Velg
Slutt sub
Privat funksjon GetCheckedRadioButton (_
ByVal Parent As Control) _
Som RadioButton
Dim FormControl As Control
Dim RB Som RadioButton
For hver FormControl In Parent. kontroller
Hvis FormControl. GetType () Er GetType (RadioButton) da
RB = DirectCast (FormControl, RadioButton)
Hvis RB.Sjekket, returner deretter RB
Slutt om
neste
Returner ingenting
Sluttfunksjon

Koden i handling ser slik ut:


Klikk her for å vise illustrasjonen
Klikk på Tilbake-knappen i nettleseren for å gå tilbake