![]() |
(const Param:Integer)oder(Param:Integer)-Unterschied?
Ich würde gerne mal wissen ,wo genau liegt die Unterschied zwischen zwei Deklarationen:
Delphi-Quellcode:
und
// Fall 1
function MyFunc(const Param:Integer):Boolean;
Delphi-Quellcode:
Soviel ich weiss, im Fall 1 kann man die "Param" Variable in der Funktion nicht ändern, sondern nur deren Wert benutzen.
// Fall 2
function MyFunc(Param:Integer):Boolean; Im Fall 2 wird eine Lokale Kopie von Param angelegt, und mir der kann man dann alles machen, was man sonst mit einer Variavle machen kann, hat aber keine Auswirkungen nach aussen. Gibt es noch welche Unterschiede? Und wo ist der Sinn der Sache, const oder nicht const ? |
Re: (const Param:Integer)oder(Param:Integer)-Unterschied?
Übergabe:
Fall 1: Referenz Fall 2: Wert |
Re: (const Param:Integer)oder(Param:Integer)-Unterschied?
Delphi-Quellcode:
ist das gleiche wie
function MyFunc(const Param:Integer):Boolean;
Delphi-Quellcode:
mit dem großen Unterschied das bei "const" der Wert nicht geändert werden darf/kann.
function MyFunc(var Param:Integer):Boolean;
|
Re: (const Param:Integer)oder(Param:Integer)-Unterschied?
out hast du dann vergessen ;) ... das gehört auch in diese Reihe. In C hingegen ist OUT nur ein leeres Makro.
|
Re: (const Param:Integer)oder(Param:Integer)-Unterschied?
genau, out gibt es auch noch.
Einen Nachtrag noch zu dem Const. Wenn du folgendes versuchst
Delphi-Quellcode:
meckert der Compiler eben weil es ein Constanter Wert ist.
function MyFunc(const Param:Integer):Boolean;
begin Param := 9; end; wenn du aber das ganze so machst
Delphi-Quellcode:
dann meckert kein Compiler und du siehst das der Übergabewert sich ändert.
function MyFunc(const Param:Integer):Boolean;
begin PInteger(@Param)^ := 9; end; wenn du also zweite variante wie folgt aufrufst
Delphi-Quellcode:
und SpinEdit1.Value = 5 ist, dann
var test1: Integer;
begin test1 := SpinEdit1.Value; ShowThat(test1); showmessage(inttostr(test1)); wirst du bei deinem ShowMessage dann sehen das test1 auf einmal 5 ist weil das const überlistet wurde. Und genau dafür ist Const eigentlich da, damit sowas nicht passiert. |
Re: (const Param:Integer)oder(Param:Integer)-Unterschied?
Zitat:
Das const hat bei Integer nur die Wirkung, dass der Compiler den Parameter nicht verändern lässt. Er wird aber immernoch mit ByValue übergeben. const hat hauptsächlich bei Strings eine besondere Bedeutung. Fehlt es dort nämlich, muss der Compiler einen try/finally Block aufbauen, der die Referenzzählung des Strings verwaltet. Ansonsten könnte man ja den String nach außen hin verändern (Seiteneffekt). Und genau hier ist das const eine Optimierung, die einiges an Geschwindigkeit aus so manchen Programmen herausholen kann. Probiert mal folgenden Code einmal ohne "const" und dann mit aus. Da braucht man nicht mal einen Profiler um den Geschwindigkeitsunterschied feststellen zu können. Ein Breakpoint
Delphi-Quellcode:
Vorausgesetzt ist natürlich immer, dass man die String-Parameter in der Funktion auch nicht ändert, also ein const überhaupt möglich ist.
procedure Stupid({const} S: string);
begin Length(S); // damit da nichts wegoptimiert wird end; procedure DoSomething; var A: string; i: Integer; begin A := 'Hallo'; A := A + ' du da'; // damit A keinen Konstanten-Referenzzähler hat (=-1) for i := 0 to 50000000 do Stupid(A); end; initialization DoSomething; // hier Breakpoint setzen (F5), Programm staten (F9) und drüber steppen (F8) end. |
Re: (const Param:Integer)oder(Param:Integer)-Unterschied?
ich kenn den unterschied, ich wollte damit nur die wirkungsweise von const demonstrieren, nämlich das eine referenz übergeben wird und keine kopie, und das konnte man wunderbar demonstrieren in dem man das const "ignoriert" und den wert innerhalb der funktion ändert.
|
Re: (const Param:Integer)oder(Param:Integer)-Unterschied?
Zitat:
Fall 1:
Delphi-Quellcode:
Fall 2:
MyFunc(3);//<-OK
Delphi-Quellcode:
MyFunc(3);//<-Fehler
|
Re: (const Param:Integer)oder(Param:Integer)-Unterschied?
Zitat:
Zitat:
Code:
Bei einem CallByRef müsste der Code so aussehen:
mov eax, [MyInteger] // Wert von MyInteger in eax (=1. Parameter bei register-Call)
call DoSomeWildThings
Code:
Oder alternativ:
mov eax, OFFSET MyInteger // Adresse von MyInteger in eax (=1. Parameter bei register-Call)
call DoSomeWildThings
Code:
lea eax, MyInteger // Adresse von MyInteger in eax (=1. Parameter bei register-Call)
call DoSomeWildThings Bei Arrays und Records stimmt deine Aussage. Eine Faustregel ist: Alles was sich in 4 Bytes reinquetschen lässt, wird bei const als Kopie auf den Stack übergeben. Alles andere geht als Zeiger auf den Stack. Zitat:
|
Re: (const Param:Integer)oder(Param:Integer)-Unterschied?
Komich, als ich aber per Pointer zugriff in der Funktion den Integer der mit Const übergeben wurde geändert hab wurde auch der Original-Integer geändert. Wohlmöglich sehen bei dir die Assemblerbefehle nur anders aus weil in deinem Beispiel etwas optimiert wird. Wenn ich nicht über den Umweg von SpinEdit1.Value sondern den Value fest in den Source codiere dann klappt das auch nicht weil dann durch die optimierung der Wert direkt übergeben wird ohne die Variable.
|
Alle Zeitangaben in WEZ +1. Es ist jetzt 21:53 Uhr. |
Powered by vBulletin® Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz