私は以下のようなレコードと内部ダイナミック配列を持つN×Mの行列(クラス)を実装しています。R1:= R2のレコードのディープコピー、またはレコードのあるNxMマトリックスを実装する良い方法はありますか?
TMat = record
public
// contents
_Elem: array of array of Double;
//
procedure SetSize(Row, Col: Integer);
procedure Add(const M: TMat);
procedure Subtract(const M: TMat);
function Multiply(const M: TMat): TMat;
//..
class operator Add(A, B: TMat): TMat;
class operator Subtract(A, B: TMat): TMat;
//..
class operator Implicit(A: TMat): TMat; // call assign inside proc.
// <--Self Implicit(which isn't be used in D2007, got compilation error in DelphiXE)
procedure Assign(const M: TMat); // copy _Elem inside proc.
// <-- I don't want to use it explicitly.
end;
レコードを選択するのは、作成/使用/割り当てを使用しないためです。
動的配列では、M1.Assign(M2)の代わりにM1:= M2で値を(深く)コピーすることはできません。
私は自己暗黙の変換方法を宣言しようとしましたが、M1:= M2では使用できません。
(暗黙(のconstのpA:PMAT):TMATとM1:= @ M2は動作しますが、それはかなり醜いと読めないのです...)
レコードの割り当てをフックする方法はありますか?
レコードがあるN×M行列を実装する方法はありますか?
ありがとうございます。
編集:
私はバリーの方法で以下のように実装され、正常に動作を確認しました。
type
TDDArray = array of array of Double;
TMat = record
private
procedure CopyElementsIfOthersRefer;
public
_Elem: TDDArray;
_FRefCounter: IInterface;
..
end;
procedure TMat.SetSize(const RowSize, ColSize: Integer);
begin
SetLength(_Elem, RowSize, ColSize);
if not Assigned(_FRefCounter) then
_FRefCounter := TInterfacedObject.Create;
end;
procedure TMat.Assign(const Source: TMat);
var
I: Integer;
SrcElem: TDDArray;
begin
SrcElem := Source._Elem; // Allows self assign
SetLength(Self._Elem, 0, 0);
SetLength(Self._Elem, Length(SrcElem));
for I := 0 to Length(SrcElem) - 1 do
begin
SetLength(Self._Elem[I], Length(SrcElem[I]));
Self._Elem[I] := Copy(SrcElem[I]);
end;
end;
procedure TMat.CopyElementsIfOthersRefer;
begin
if (_FRefCounter as TInterfacedObject).RefCount > 1 then
begin
Self.Assign(Self); // Self Copy
end;
end;
私はそれが効率的ではないと同意します。純粋なレコードで割り当てを使用するだけで、絶対に高速です。
しかし、それは。かなり便利で、より読みやすいです(と面白い。:-)
私はそれが軽い計算やプリプロダクションのプロトタイピングに便利だと思います。ではない ?
EDIT2:
kibabは、動的配列自体の機能になっ参照カウントを与えます。
バリーのソリューションは内部のインプラントから独立しており、今後の64ビットコンパイラでは何も変更することなく動作するかもしれませんが、このケースでは効率が良いのでキバブを好むのが&です。ありがとう。
TMat = record
private
procedure CopyElementsIfOthersRefer;
public
_Elem: TDDArray;
..
end;
procedure TMat.SetSize(const RowSize, ColSize: Integer);
begin
SetLength(_Elem, RowSize, ColSize);
end;
function GetDynArrayRefCnt(const ADynArray): Longword;
begin
if Pointer(ADynArray) = nil then
Result := 1 {or 0, depending what you need}
else
Result := PLongword(Longword(ADynArray) - 8)^;
end;
procedure TMat.CopyElementsIfOthersRefer;
begin
if GetDynArrayRefCnt(_Elem) > 1 then
Self.Assign(Self);
end;
+1興味深い質問です。私は演算子のオーバーロードを伴うレコードの値の型に固執すると思います。 –