微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

delphi – 具有R1:= R2的记录的深度拷贝,或者是否有很好的方法来实现NxM矩阵与记录?

我正在实现一个N x M矩阵(类)与一个记录和一个内部动态数组,如下所示.
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)= M2而不是M1.Assign(M2)复制值.

我尝试声明自我隐式转换方法,但不能用于M1:= M2.

(Implicit(const pA:PMat)):TMat和M1:= @ M2工作,但它是非常丑陋和不可读的..)

有没有办法挂钩记录的任务?

还是有建议用记录来实现N×M矩阵?

提前致谢.

编辑:

我用Barry的方法实施如下,确认工作正常.

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);
  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给出函数获取动态数组本身的引用计数.

Barry的解决方案更加独立于内部的内容,也许对即将到来的64位编译器没有任何修改,但在这种情况下,我更喜欢kibab的简单性和效率.谢谢.

TMat = record
  private
     procedure copyElementsIfOthersRefer;
  public
    _Elem: TDDArray;
   ..
  end;

procedure TMat.SetSize(const 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;

解决方法

您可以使用记录中的接口字段引用来确定数组是否由多个记录共享:只需检查接口后面的对象的引用计数,并且知道数组中的数据是共享的.这样,您可以在修改后懒洋洋地复制,但是当矩阵未被修改时仍然使用数据共享.

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


 从网上看到《Delphi API HOOK完全说明》这篇文章,基本上都是大家转来转去,原文出处我已经找不到了。这篇文章写的很不错,但最后部分“PermuteFunction 的终极版本”描述的不太清楚,完全按照该文章代码执行,是不行的。可能是作者故意这样做的?本文最后提供修正后的下载地址。原文如下:一、关于API Hook1.什么是API Hook不知道大家是否还记得,在DO
  从网上看到《Delphi API HOOK完全说明》这篇文章,基本上都是大家转来转去,原文出处我已经找不到了。 这篇文章写的很不错,但最后部分“PermuteFunction 的终极版本”描述的不太清楚,完全按照该文章代码执行,是不行的。需要修改mess.pas中代码才行。其实文中提到的一个结构,代码中并没有使用typePIMAGE_IMPORT_DESCRIPTOR = ^IMA
ffmpeg 是一套强大的开源的多媒体库 一般都是用 c/c++ 调用, 抽空研究了一下该库的最新版 ,把部分api 翻译成了dephi版的 记录一下 地址 ffmpegvcl.zip
32位CPU所含有的寄存器有:4个数据寄存器(EAX、EBX、ECX和EDX)2个变址和指针寄存器(ESI和EDI) 2个指针寄存器(ESP和EBP)6个段寄存器(ES、CS、SS、DS、FS和GS)
1 mov dst, src dst是目的操作数,src是源操作数,指令实现的功能是:将源操作数送到目的操作数中,即:(dst) &lt;--(src) 1.dst和src类型必须匹配,即必须同为字节
有三个API函数可以运行可执行文件WinExec、ShellExecute和CreateProcess。 1.CreateProcess因为使用复杂,比较少用。 2.WinExec主要运行EXE文件。如:WinExec('Notepad.exe Readme.txt', SW_SHOW); 3.ShellExecute不仅可以运行EXE文件,也可以运行已经关联的文件。 首先必须引用shellapi
API原型: Declare Function MoveFileEx& Lib "kernel32" Alias "MoveFileExA" (ByVal lpExistingFileName As String, ByVal lpNewFileName As String, ByVal dwFlags As Long) 参数 类型及说明 lpExistingFileName String,欲移
附带通用控件安装方法: ---------- 基本安装 1、对于单个控件,Componet-->install component..-->PAS或DCU文件-->install; 2、对于带*.dpk文件的控件包,File-->Open(下拉列表框中选*.dpk)-->install即可; 3、对于带*.bpl文件的控件包,Install Packages-->Add-->bpl文件名即可; 4
type   TRec=Record     msg:string;     pic:TMemoryStream; end; procedure TForm2.BitBtn1Click(Sender: TObject); var   ms:TMemoryStream;   Rec1,Rec2:TRec;   cc:tmemorystream;   jpg:TJPEGImage; begin   R