Um überhaupt vergleichen zu können, würde ich erstmal die interpolationen in TGPGraphic abstellen. Dann die Frage warum überhaupt
GDI+? Wenn du ein so hochauflösendes Bild hast, kannst du nicht auch SetStretchBLTMode HalfTone zum runterskalieren nehmen?
Benutz ich z.B. in meinem AsciiImage:
TGDIRenderContext.EndScene(Zeile 135)
Und so schauts aus:
AsciiImage for Delphi: GDI-Downsampling, FireMonkey and more!
EDIT: Vergiss nicht, dass du da bei
GDI+ mitunter unsäglich viel konvertierst(Kann dir auch bei
GDI passieren). Wenn die Pixel-Formate zwischen Source und Target nicht gleich sind, wird die source immer konvertiert beim Zeichnen! IIRC solltest du am besten mit 32bit Grafiken arbeiten.
zu: "würde ich erstmal die interpolationen in TGPGraphic abstellen"
ich hab alle Modi probiert:
InterpolationModeInvalid 270 ms
InterpolationModeDefault 270 ms
InterpolationModeLowQuality 270 ms
InterpolationModeHighQuality 325 ms
InterpolationModeBilinear 270 ms
InterpolationModeBicubic 620 ms
InterpolationModeNearestNeighbor 180 ms
InterpolationModeHighQualityBilinear 370 ms
InterpolationModeHighQualityBicubic 320 MS
Alle Zeiten für einmal zeichnen.
Dagegen StretchBlt 2 ms
Bringt also die benötigte Zeit nicht einmal ansatzweise in die Zeit für StretchBlt.
Und warum bin ich so spitz auf TGPBitmap bin, statt TBitmap?
Das Bild liegt idR als .jpg vor, kann aber auch als .png oder .tif oder .bmp sein.
Bisher habe ich z.B. .jpg entsprechend der untenstehenden "LoadBmpViaJPEG" geladen.
Zur Zeit benutze ich die untenstehende "LoadBmpViaGPBitmap"
Und dann hab ich die untenstehende "LoadGPBitmap" getestet.
Die jeweils benötigten Zeiten:
"LoadBmpViaJPEG" : 970 ms
"LoadBmpViaGPBitmap" 270 ms
"LoadGPBitmap" 319 µs
Da ich das Bild nur ausgeben kann, wenn ich es vorher lade, drängt sich das laden in eine TGPBitmap förmlich auf.
Deshalb möchte auch bei allem was folgt am liebsten mit TGPBitmap weitermachen.
Delphi-Quellcode:
FUNCTION LoadBmpViaJPEG(const Dsn:String):Int64;
var Jpg:TJpegImage; Bmp:TBitmap;
begin
Result:=TimeStamp;
Jpg:=TJPEGImage.Create;
Bmp:=TBitmap.Create;
Jpg.LoadFromFile(Dsn);
Bmp.Assign(Jpg);
Jpg.Free;
Result:=TimeStamp-Result;
Bmp.Free;
end;
Delphi-Quellcode:
FUNCTION LoadBmpViaGPBitmap(const Dsn:String):Int64;
var W,H:Integer; Image:TGPBitmap; Graphics:TGPGraphics; Bmp:TBitmap;
begin
Result:=TimeStamp;
Bmp:=TBitmap.Create;
Image:=TGPBitmap.Create(Dsn);
W:=Image.GetWidth;
H:=Image.GetHeight;
Bmp.SetSize(W,H);
Graphics:=TGPGraphics.Create(Bmp.Canvas.Handle);
Graphics.SetInterpolationMode(InterpolationModeHighQualityBicubic);
Graphics.DrawImage(Image,MakeRect(0,0,W,H));
Graphics.Free;
Image.Free;
Result:=TimeStamp-Result;
Bmp.Free;
end;
Delphi-Quellcode:
FUNCTION LoadGPBitmap(const Dsn:String):Int64;
var I:Integer; Image:TGPBitmap;
begin
Result:=TimeStamp;
Image:=TGPBitmap.Create(Dsn);
Result:=TimeStamp-Result;
Image.Free;
end;
Zur Vollständigkeit:
Delphi-Quellcode:
FUNCTION TimeStamp:Int64;
asm
rdtsc
{$IFDEF CPUX64}
shl rdx,32
or rax,rdx
{$ENDIF}
end;