关于Delphi中的数值类型问题~~(100分)

  • 主题发起人 主题发起人 zzandyzh
  • 开始时间 开始时间
Z

zzandyzh

Unregistered / Unconfirmed
GUEST, unregistred user!
现有以下几个问题,希望大家能帮助!
1.Delphi中如何表示无符号的64位整数?
2.如何把一个字符串转换成无符号的64位整数?
3.C语言中的LPTSTR类型在Delphi中如何定义,比如有一个C写的DLL,函数定义是这样地:
MakeDiskImage(UINT64 u64Bytes, LPTSTR lpBad),在Delphi中怎么调用啊?
4.下面是一个C定义的一个结构体,在Delphi中怎么定义!
typedef struct My_p
{
INTEGER liBytes;
LARGE lizeBytes;
BYTE bPpe;
My_p *pNext;
};

谢谢各位大侠给看看!
 
第一和第二可以用这个单元解决:
unit UInt64Lib;
{
Unsigned 64 bit Integer support for Delphi 6 and Kylix.
Copyright (c) 2002 by DelphiFactory Netherlands BV

This unit provides to routines to convert between strings and unsigned
64 bit integers.

UInt64 range: 0..18446744073709551615

Thanks to J.H.Plantenberg for the assembler part.

For comments mail to:
erwin@delphi-factory.com
}

interface

resourcestring
SInvalidUInt64 = '''%s'' is not a valid UInt64 value';

{ The UInt64 type is declared as a strong typed Int64, since both compilers
don't support unsigned 64 bit integers. This way you can at least do some
calculations. }
type
UInt64 = type Int64;

{ StrToUInt64 converts the given string to an unsigned 64 bit integer value.
If the string doesn't contain a valid value, an EConvertError exception is
raised. }
function StrToUInt64(const S: AnsiString): UInt64;

{ UInt64ToStr converts the given value to its decimal string representation. }
function UInt64ToStr(Value: UInt64): string;
function UInt64ToHex(Value: UInt64): string;

implementation

// For speed we are going to use the fact that long strings are
// guaranteed to be null-terminated:
{$R-} // Range checking must be disabled
{$B-} // Do not perform complete boolean expression evaluations

uses
SysUtils;

type
_UInt64 = packed record // Split the 64 bit into 2x32 bit
Lo, Hi : LongWord;
end;

procedure RaiseEConvertError(const S: AnsiString);
begin
// raise an exception explaining the input string was invalid
raise EConvertError.CreateResFmt(@SInvalidUInt64, );
end;

function StrToUInt64(const S: AnsiString): UInt64;
var
I: LongWord;
Dig: Integer;
Digit : LongWord;
Hi,Lo : LongWord;

begin
// check if S is empty (is nil pointer)
if S = '' then
RaiseEConvertError(S);

// start at the first character
I := 1;

// trim leading spaces
while S = ' ' do
Inc(I);

if S = '-' then // check for minus sign
RaiseEConvertError(S);

if S = '+' then // check for plus sign
Inc(I);

// Check for hexidecimal string id: '$' or '0x'
if (S = '$') or ((S = '0') and (Upcase(S[I+1]) = 'X')) then
begin
// trim leading zero (if '0x' hex marker)
if S = '0' then
Inc(I);

// trim hex marker
Inc(I);

// Check if empty
if S = #0 then
RaiseEConvertError(S);

// init loop
Dig := 0;
Result := 0;

// while not end of string
while S <> #0 do
begin
// try character convert
case S of
'0'..'9': Dig := Ord(S) - Ord('0');
'A'..'F': Dig := Ord(S) - (Ord('A') - 10);
'a'..'f': Dig := Ord(S) - (Ord('a') - 10);
else
RaiseEConvertError(S)
end;

// still enough room in result?
if Result shr 60 > 0 then
RaiseEConvertError(S);

// shift converted digit into result
Result := Result shl 4 + Dig;

// next char
Inc(I);
end;
end
else begin // decimal unsigned 64 bit conversion

// check if not empty
if S = #0 then
RaiseEConvertError(S);

Hi := 0;
Lo := 0;
while S <> #0 do
begin
// Extract the digit from the string and convert it ASCII->Byte
Digit := Ord(S) xor Ord('0');

// Some assembler to perform an unsigned 64 bit integer calculation.
// This asm code runs in D6 and Kylix (PIC code).
// HiLo := (HiLo*10)+Digit
asm
push esi // save register

// calculate: Hi * 10;
mov eax, Hi // Load Hi
mov ecx, 10 // multiplier is 10
mul ecx // EDX:EAX := EAX*ECX
or edx, edx // Overflow?
jnz @TooBig // yes -> bye bye

// calculate: Lo * 10
mov esi, eax // save Hi value
mov eax, Lo // load Lo
mul ecx // EDX:EAX := EAX*ECX

// Combine Hi, Lo, and overflow of Lo to form HiLo result
add edx, esi // EDX:EAX := HiLo*10

// HiLo := HiLo + Digit
add eax, Digit // EDX:EAX := HiLo+Digit
adc edx, 0 // check overflow
jc @TooBig // yes -> bye bye

// save HiLo
mov Hi, edx // Hi := EDX
mov Lo, eax // Lo := EAX
jmp @TheEnd // successfull finish
@TooBig:
mov Digit, 666 // something went wrong: invalidate Digit
@TheEnd:

pop esi // restore register
end;

// Check if digit was legal and if the previous calculation was a success
if not (Digit in [0..9]) then
RaiseEConvertError(S);

// proceed to the next digit
Inc(I);
end;
// Return HiLo as an unsigned 64 bit integer
_UInt64(Result).Lo := Lo;
_UInt64(Result).Hi := Hi;
end;
end;


function UInt64ToStr(Value: UInt64): string;
const
BiggestUInt64Str = '18446744073709551615';
MaxBCD = Length(BiggestUInt64Str);

type
TBCD = array[0..MaxBCD-1] of Integer; { Index 0 is highest BCD digit}

procedure AddBCD(var BCD : TBCD; Pos,Value : Integer);
begin
Inc(BCD[Pos], Value);
if BCD[Pos] > 9 then
begin
Dec(BCD[Pos],10);
AddBCD(BCD,Pos-1, 1);
end;
end;

procedure IncBCD(var A : TBCD; const B : TBCD);
var
I : Integer;
begin
for I := MaxBCD-1 downto 0 do
AddBCD(A, I, B);
end;

var
ValueBCD : TBCD;
BitValue : TBCD;
Tmp : TBCD;
I : Integer;
Ofs : Integer;

begin
// default to zero
FillChar(ValueBCD, SizeOf(ValueBCD), 0);

// set bit value BCD. Lowest bit has value 1
FillChar(BitValue, SizeOf(BitValue), 0);
BitValue[MaxBCD-1] := 1;

// check if there are bits available
while Value <> 0 do
begin
// if current lowest bit is set
// Increment the BCD value with the current bit value
if Value and 1 <> 0 then
IncBCD(ValueBCD, BitValue);

// proceed to the next bit
Value := Value shr 1;

// Double the BitValue
Tmp := BitValue;
IncBCD(BitValue, Tmp);
end;

// Find highest non zero decimal
Ofs := 0;
while (Ofs < MaxBCD) and (ValueBCD[Ofs] = 0) do
Inc(Ofs);

// check if any non zero decimals present
if Ofs < MaxBCD then
begin
// convert BCD result to ASCII result
SetLength(Result, MaxBCD-Ofs);
I := Ofs;
Repeat
Result[I-Ofs+1] := Char(ValueBCD+Ord('0'));
Inc(I);
until I > MaxBCD-1;
end
else
Result := '0'; // nothing set -> value is '0'
end;

function UInt64ToHex(Value: UInt64): string;
type
TByteArray = array[0..7] of Byte;
var
I : Byte;
B1 : ^TByteArray;
begin
Result := '';
//Place the pointer of bytes to the value (-:
B1 := Addr(Value);

for I := High(B1^) downto Low(B1^) do
Result := Result + IntToHex(B1^, 2);
while ((Result[1] = '0') and (Length(Result) > 1)) do
Delete(Result, 1, 1);
end;


end.

第三个问题,可结合上面的单元声明为
procedure MakeDiskImage(u64Bytes: UInt64; lpBad: PChar); stdcall;
最后一个问题
type
PMy_p = ^TMy_p;
TMy_p = record
liBytes: Integer;
lizeBytes: UInt64;
bPpe: Byte;
pNext: PMy_p;
end;
 
3
LPTSTR 对应的是PAnsiChar 或者PWideChar
你问问他的编译选项,要是定义了unicode就是PWideChar 否则是PAnsiChar
4你需要知道 那个结构体有前两个类型是定义
 
后退
顶部