gpt4 book ai didi

delphi - 尝试使用 Lazarus 和 Freepascal 创建数据传输率计算器

转载 作者:行者123 更新时间:2023-12-01 20:09:25 26 4
gpt4 key购买 nike

我正在尝试向我的应用程序界面添加“传输速率:XGb p\min”功能。我使用 Lazarus 1.2.2 和 Freepascal 2.6.4 以毫秒级计算。

我有一个循环,它从磁盘读取 64Kb block ,对每个 64Kb block 进行填充,然后重复,直到读取整个磁盘或直到用户单击中止按钮。我试图计算每次读取 64Kb 需要多长时间,然后计算每分钟读取数据的平均速率,即“每分钟 3Gb”。

我有一个名为“GetTimeInMilliseconds”的自定义函数,它可以将标准时间计算为毫秒。

此外,为了避免接口(interface)刷新部分毫秒,我有一个计数器,试图确保接口(interface)仅在循环中每 50 次刷新一次,即当 64Kb block 被读取 50 次时,计数器就会重置到 0。

问题是,传输显示要么没有填充,要么填充了不准确的数字,例如 RAID0 设备的“234Mb p\min”!有时,它会变得更实际,例如“3.4Gb p\min”。如果在同一台 PC 和同一磁盘上重复运行,它应该始终准确,而不是不一致的不准确。

这是我的 Try...finally 循环,它执行循环。它还称为 FormatByteSize,这是一个自定义函数,用于计算整数大小并将其转换为 XMb、XGb、XTb 等。

var

ProgressCounter, BytesTransferred, BytesPerSecond : integer;

Buffer : array [0..65535] of Byte; // 65536, 64Kb buffer

ctx : TSHA1Context;

Digest : TSHA1Digest;

NewPos, ExactDiskSize, SectorCount, TimeStartRead, TimeEndRead,
MillisecondsElapsed, BytesPerMillisecond, BytesPerMinute : Int64;

StartTime, EndTime, TimeTaken : TDateTime;
...
begin
... // Various stuff related to handles on disks etc
try
SHA1Init(ctx);
FileSeek(hSelectedDisk, 0, 0);
repeat
ProgressCounter := ProgressCounter + 1; // We use this update the progress display occasionally, instead of every buffer read
TimeStartRead := GetTimeInMilliSeconds(Time); // Starting time, in Ms

// The hashing bit...

FileRead(hSelectedDisk, Buffer, 65536); // Read 65536 = 64kb at a time
NewPos := NewPos + SizeOf(Buffer);
SHA1Update(ctx, Buffer, SizeOf(Buffer));
FileSeek(hSelectedDisk, NewPos, 0);
lblBytesLeftToHashB.Caption := IntToStr(ExactDiskSize - NewPos) + ' bytes, ' + FormatByteSize(ExactDiskSize - NewPos);

// End of the hashing bit...

TimeEndRead := GetTimeInMilliSeconds(Time);; // End time in Ms
MillisecondsElapsed := (TimeEndRead - TimeStartRead); // Record how many Ms's have elapsed

// Only if the loop has gone round a while (50 times?), update the progress display
if ProgressCounter = 50 then
begin
if (TimeStartRead > 0) and (TimeEndRead > 0) and (MillisecondsElapsed > 0) then // Only do the divisions and computations if all timings have computed to a positive number
begin
BytesTransferred := SizeOf(Buffer);
BytesPerMillisecond := BytesTransferred DIV MilliSecondsElapsed; // BytesPerMillisecond if often reported as zero, even though BytesTRansferred and MilliSecondsElapsed are valid numbers? Maybe this is the fault?
BytesPerSecond := BytesPerMillisecond * 1000; // Convert the bytes read per millisecond into bytes read per second
BytesPerMinute := BytesPerSecond * 60; // Convert the bytes read per second into bytes read per minute
lblSpeedB.Caption := FormatByteSize(BytesPerMinute) + ' p\min'; // now convert the large "bytes per minute" figure into Mb, Gb or Tb

// Reset the progress counters to zero for another chunk of looping
ProgressCounter := 0;
BytesPerMillisecond := 0;
BytesPerSecond := 0;
BytesPerMinute := 0;
ProgressCounter := 0;
end;
end;
Application.ProcessMessages;
until (NewPos >= ExactDiskSize) or (Stop = true); // Stop looping over the disk
// Compute the final hash value
SHA1Final(ctx, Digest);
lblBytesLeftToHashB.Caption:= '0';
finally
// The handle may have been released by pressing stop. If not, the handle will still be active so lets close it.
if not hSelectedDisk = INVALID_HANDLE_VALUE then CloseHandle(hSelectedDisk);
EndTime := Now;
TimeTaken := EndTime - StartTime;
lblEndTimeB.Caption := FormatDateTime('dd/mm/yy hh:mm:ss', EndTime);
if not stop then edtComputedHash.Text := SHA1Print(Digest);
Application.ProcessMessages;
...

Function GetTimeInMilliSeconds(theTime : TTime): Int64;
// http://www.delphipages.com/forum/archive/index.php/t-135103.html
var
Hour, Min, Sec, MSec: Word;
begin
DecodeTime(theTime,Hour, Min, Sec, MSec);
Result := (Hour * 3600000) + (Min * 60000) + (Sec * 1000) + MSec;
end;

示例:

Form1.Caption := IntToStr( GetTimeInMilliSeconds(Time) );

最佳答案

我觉得你让这件事变得比需要的复杂得多。

将文件处理代码移至单独的线程中。在该代码中,当您处理完文件中的字节时,请使用 InterlockedAdd 来记录已处理的字节总数。

在 GUI 线程中,使用某种计时器,使用 InterlockedExchange 读取 InterlockedAdd 已修改的值并将其重置为零。然后计算自上次计时器计时以来已经过去的时间量。给定在耗时内处理的字节总数,计算每分钟的字节数并更新您的 GUI。

由于文件操作是在单独的线程上进行的,因此您无需担心更新 GUI 的频率(尽管这样做太频繁会浪费时间)。

如果您想计算整个操作中每分钟的平均字节数,请不要在每个周期重置总字节计数器。

关于delphi - 尝试使用 Lazarus 和 Freepascal 创建数据传输率计算器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24331247/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com