Вот несколько функций для операций с двухмерными массивами. Самый простой путь для создания собственной библиотеки. Процедуры SetV и GetV позволяют читать и сохранять элементы массива V
Array (его Вы можете объявить как угодно).
Например:
Код type
VArray : Array [1..1] of double;
var
X : ^VArray ;
NR, NC : Longint;
begin
NR := 10000;
NC := 100;
if AllocArray (pointer(X), N*Sizeof(VArray )) then exit;
SetV(X^, NC, 2000, 5, 3.27); { X[2000,5] := 3.27 }
end;
function AllocArray (var V : pointer; const N : longint) : Boolean;
begin {распределяем память для массива V размера N}
try
GetMem(V, N);
except
ShowMessage('ОШИБКА выделения памяти. Размер:' + IntToStr(N));
Result := True;
exit;
end;
FillChar(V^, N, 0); {в случае включения длинных строк заполняем их нулями}
Result := False;
end;
procedure SetV(var X : Varray ;const N,ir,ic : LongInt;const value :
double);
begin {заполняем элементами двухмерный массив X размером ? x N : X[ir,ic] := value}
X[N*(ir-1) + ic] := value;
end;
function GetV(const X : Varray ; const N, ir,ic : Longint) : double;
begin {возвращаем величины X[ir,ic] для двухмерного массива шириной N столбцов}
Result := X[N*(ir-1) + ic];
end;
Самый простой путь - создать массив динамически
Код Myarray := GetMem(rows * cols * sizeof(byte,word,single,double и пр.)
сделайте функцию fetch_num типа
Код function fetch_num(r,c:integer) : single;
result := pointer + row + col*rows
и затем вместо my
array [2,3] напишите
Код myarray .fetch_num(2,3)
Вот способ создания одно- и двухмерных динамических массивов:
Код (*
модуль для создания двух очень простых классов обработки динамических массивов
TDynaArray : одномерный массив
TDynaMatrix : двумерный динамический массив
*)
unit DynArray ;
INTERFACE
uses
SysUtils;
Type
TDynArray BaseType = double;
Const
vMaxElements = (High(Cardinal) - $f) div sizeof(TDynArray BaseType);
{= гарантирует максимально возможный массив =}
Type
TDynArray NDX = 1..vMaxElements;
TArray Elements = array [TDynArray NDX] of TDynArray BaseType;
{= самый большой массив TDynArray BaseType, который мы может объявить =}
PArray Elements = ^TArray Elements;
{= указатель на массив =}
EDynArray RangeError = CLASS(ERangeError);
TDynArray = CLASS
Private
fDimension : TDynArray NDX;
fMemAllocated : word;
Function GetElement(N : TDynArray NDX) : TDynArray BaseType;
Procedure SetElement(N : TDynArray NDX; const NewValue : TDynArray BaseType);
Protected
Elements : PArray Elements;
Public
Constructor Create(NumElements : TDynArray NDX);
Destructor Destroy; override;
Procedure Resize(NewDimension : TDynArray NDX); virtual;
Property dimension : TDynArray NDX
read fDimension;
Property Element[N : TDynArray NDX] : TDynArray BaseType
read GetElement
write SetElement;
default;
END;
Const
vMaxMatrixColumns = 65520 div sizeof(TDynArray );
{= построение матрицы класса с использованием массива объектов TDynArray =}
Type
TMatrixNDX = 1..vMaxMatrixColumns;
TMatrixElements = array [TMatrixNDX] of TDynArray ;
{= каждая колонка матрицы будет динамическим массивом =}
PMatrixElements = ^TMatrixElements;
{= указатель на массив указателей... =}
TDynaMatrix = CLASS
Private
fRows : TDynArray NDX;
fColumns : TMatrixNDX;
fMemAllocated : longint;
Function GetElement( row : TDynArray NDX;
column : TMatrixNDX) : TDynArray BaseType;
Procedure SetElement( row : TDynArray NDX;
column : TMatrixNDX;
const NewValue : TDynArray BaseType);
Protected
mtxElements : PMatrixElements;
Public
Constructor Create(NumRows : TDynArray NDX; NumColumns : TMatrixNDX);
Destructor Destroy; override;
Property rows : TDynArray NDX
read fRows;
Property columns : TMatrixNDX
read fColumns;
Property Element[row : TDynArray NDX; column : TMatrixNDX] : TDynArray BaseType
read GetElement
write SetElement;
default;
END;
IMPLEMENTATION
(*
методы TDynArray
*)
Constructor TDynArray .Create(NumElements : TDynArray NDX);
BEGIN {==TDynArray .Create==}
inherited Create;
fDimension := NumElements;
GetMem( Elements, fDimension*sizeof(TDynArray BaseType) );
fMemAllocated := fDimension*sizeof(TDynArray BaseType);
FillChar( Elements^, fMemAllocated, 0 );
END; {==TDynArray .Create==}
Destructor TDynArray .Destroy;
BEGIN {==TDynArray .Destroy==}
FreeMem( Elements, fMemAllocated );
inherited Destroy;
END; {==TDynArray .Destroy==}
Procedure TDynArray .Resize(NewDimension : TDynArray NDX);
BEGIN {TDynArray .Resize==}
if (NewDimension < 1) then
raise EDynArray RangeError.CreateFMT('Индекс вышел за границы диапазона : %d', [NewDimension]);
Elements := ReAllocMem(Elements, fMemAllocated, NewDimension*sizeof(TDynArray BaseType));
fDimension := NewDimension;
fMemAllocated := fDimension*sizeof(TDynArray BaseType);
END; {TDynArray .Resize==}
Function TDynArray .GetElement(N : TDynArray NDX) : TDynArray BaseType;
BEGIN {==TDynArray .GetElement==}
if (N < 1) OR (N > fDimension) then
raise EDynArray RangeError.CreateFMT('Индекс вышел за границы диапазона : %d', [N]);
result := Elements^[N];
END; {==TDynArray .GetElement==}
Procedure TDynArray .SetElement(N : TDynArray NDX; const NewValue : TDynArray BaseType);
BEGIN {==TDynArray .SetElement==}
if (N < 1) OR (N > fDimension) then
raise EDynArray RangeError.CreateFMT('Индекс вышел за границы диапазона : %d', [N]);
Elements^[N] := NewValue;
END; {==TDynArray .SetElement==}
(*
методы TDynaMatrix
*)
Constructor TDynaMatrix.Create(NumRows : TDynArray NDX; NumColumns : TMatrixNDX);
Var col : TMatrixNDX;
BEGIN {==TDynaMatrix.Create==}
inherited Create;
fRows := NumRows;
fColumns := NumColumns;
{= выделение памяти для массива указателей (т.е. для массива TDynArray s) =}
GetMem( mtxElements, fColumns*sizeof(TDynArray ) );
fMemAllocated := fColumns*sizeof(TDynArray );
{= теперь выделяем память для каждого столбца матрицы =}
for col := 1 to fColumns do
BEGIN
mtxElements^[col] := TDynArray .Create(fRows);
inc(fMemAllocated, mtxElements^[col].fMemAllocated);
END;
END; {==TDynaMatrix.Create==}
Destructor TDynaMatrix.Destroy;
Var col : TMatrixNDX;
BEGIN {==TDynaMatrix.Destroy;==}
for col := fColumns downto 1 do
BEGIN
dec(fMemAllocated, mtxElements^[col].fMemAllocated);
mtxElements^[col].Free;
END;
FreeMem( mtxElements, fMemAllocated );
inherited Destroy;
END; {==TDynaMatrix.Destroy;==}
Function TDynaMatrix.GetElement( row : TDynArray NDX;
column : TMatrixNDX) : TDynArray BaseType;
BEGIN {==TDynaMatrix.GetElement==}
if (row < 1) OR (row > fRows) then
raise EDynArray RangeError.CreateFMT('Индекс строки вышел за границы диапазона : %d', [row]);
if (column < 1) OR (column > fColumns) then
raise EDynArray RangeError.CreateFMT('Индекс столбца вышел за границы диапазона : %d', [column]);
result := mtxElements^[column].Elements^[row];
END; {==TDynaMatrix.GetElement==}
Procedure TDynaMatrix.SetElement( row : TDynArray NDX;
column : TMatrixNDX;
const NewValue : TDynArray BaseType);
BEGIN {==TDynaMatrix.SetElement==}
if (row < 1) OR (row > fRows) then
raise EDynArray RangeError.CreateFMT('Индекс строки вышел за границы диапазона : %d', [row]);
if (column < 1) OR (column > fColumns) then
raise EDynArray RangeError.CreateFMT('Индекс столбца вышел за границы диапазона : %d', [column]);
mtxElements^[column].Elements^[row] := NewValue;
END; {==TDynaMatrix.SetElement==}
END.
Тестовая программа для модуля DynArray
uses DynArray , WinCRT;
Const
NumRows : integer = 7;
NumCols : integer = 5;
Var
M : TDynaMatrix;
row, col : integer;
BEGIN
M := TDynaMatrix.Create(NumRows, NumCols);
for row := 1 to M.Rows do
for col := 1 to M.Columns do
M[row, col] := row + col/10;
writeln('Матрица');
for row := 1 to M.Rows do
BEGIN
for col := 1 to M.Columns do
write(M[row, col]:5:1);
writeln;
END;
writeln;
writeln('Перемещение');
for col := 1 to M.Columns do
BEGIN
for row := 1 to M.Rows do
write(M[row, col]:5:1);
writeln;
END;
M.Free;
END.