как преобразовать байтовый массив в его шестнадцатеричное представление в Delphi - PullRequest
1 голос
/ 30 июня 2009

У меня есть переменная TBytes со значением [0,0,15,15]. Как я могу преобразовать это в "00FF"?

Я не хочу использовать циклы, потому что эта логика будет использоваться в функции, требующей много времени.

(Я пытался использовать BinToHex, но не смог заставить его работать со строковой переменной.)

Спасибо и всего наилучшего,

Паван.

Ответы [ 4 ]

5 голосов
/ 30 июня 2009
// Swapping is necessary because x86 is little-endian.
function Swap32(value: Integer): Integer;
asm
  bswap eax
end;

function FourBytesToHex(const bytes: TBytes): string;
var
  IntBytes: PInteger;
  FullResult: string;
begin
  Assert(Length(bytes) = SizeOf(IntBytes^));
  IntBytes := PInteger(bytes);
  FullResult := IntToHex(Swap32(IntBytes^), 8);
  Result := FullResult[2] + FullResult[4] + FullResult[6] + FullResult[8];
end;

Если эта последняя строка выглядит немного странно, это потому, что вы запросили преобразование четырехбайтового массива в строку из четырех символов, тогда как в общем случае восемь шестнадцатеричные цифры необходимы для представления четырехбайтовое значение. Я просто предполагаю, что все ваши байтовые значения меньше 16, поэтому нужна только одна шестнадцатеричная цифра. Если ваш пример был опечаткой, просто замените две последние строки на эту:

Result := IntToHex(Swap32(IntBytes^), 8);

Кстати, ваши требования, запрещающие петли, не будут выполнены. IntToHex использует цикл внутри.

3 голосов
/ 30 июня 2009
function ByteToHex(InByte:byte):shortstring;
const Digits:array[0..15] of char='0123456789ABCDEF';
begin
 result:=digits[InByte shr 4]+digits[InByte and $0F];
end;

Example :
MyHex := ByteTohex($FF);
the result
MyHex is "FF".

MyHex := ByteTohex(255);
the result
MyHex is "FF".

MyHex := ByteTohex($55);
the result
MyHex is "55".
2 голосов
/ 30 июня 2009

Это довольно быстро и работает с любым размером массива. Это похоже на BinToHex, но вместо ожидания значений байтов 0..255, он использует только низкий клев.

procedure BinToSingleHex(Buffer, Text: PAnsiChar; BufSize: Integer);
const
  Convert: array[0..15] of AnsiChar = '0123456789ABCDEF';
var
  I: Integer;
begin
  for I := 0 to BufSize - 1 do
  begin
    Text[0] := Convert[Byte(Buffer[I]) and $F];
    Inc(Text);
  end;
end;

Ассемблер, который делает то же самое:

procedure BinToSingleHex(Buffer, Text: PAnsiChar; BufSize: Integer);assembler;
asm
        PUSH    ESI
        PUSH    EDI
        MOV     ESI,EAX
        MOV     EDI,EDX
        MOV     EDX,0
        JMP     @@1
@@0:    DB      '0123456789ABCDEF'
@@1:    LODSB
        AND     DL,AL
        AND     DL,0FH
        MOV     AL,@@0.Byte[EDX]
        STOSB
        DEC     ECX
        JNE     @@1
        POP     EDI
        POP     ESI
end;

использование:

type  THexDigit=0..15;
const ArSize=16;
var   Ar:array[0..Pred(ArSize)] of THexDigit=(0,1,2,3,4,5,6,7,8,9,8,7,6,5,4,3);
      S:Array[0..Pred(ArSize)] of AnsiChar;

BinToSingleHex(@Ar,S,Length(Ar));
WriteLn(S);
0 голосов
/ 03 июля 2009

Немного опоздал на вечеринку, но почему бы не простой справочный стол?

const
HexChars : Array[0..15] of Char = ('0','1','2','3','4','5','6','7','8','9','A','B','C','D','E','F');

Предполагая, что значения TBytes равны 0..15

Function (ABytea: TBytes): string
    begin 
      Result  := HexChars[ABytea[0]];
      Result  := Result + HexChars[ABytea[1]];
      Result  := Result + HexChars[ABytea[2]];
      Result  := Result + HexChars[ABytea[3]];
    end;

конечно, аккуратнее с циклом :) и нуждается в изменении значений байтов выше 15:

begin 
  Result  := HexChars[ABytea[0] shr 4];
  Result  := Result + HexChars[ABytea[0] and $0F];
  Result  := Result + HexChars[ABytea[1] shr 4];
  Result  := Result + HexChars[ABytea[1] and $0F];
  Result  := Result + HexChars[ABytea[2] shr 4];
  Result  := Result + HexChars[ABytea[2] and $0F];
  Result  := Result + HexChars[ABytea[3] shr 4];
  Result  := Result + HexChars[ABytea[3] and $0F];
end;

Все еще аккуратнее с петлей, особенно если TBytes становится больше

...