1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-04 18:06:22 +03:00

0030478: Foundation Classes, NCollection_Array2 - provide Resize/Move methods consistent to NCollection_Array1

This commit is contained in:
kgv 2019-02-06 12:29:21 +03:00 committed by apn
parent 597fde688e
commit 92435cd0ff
2 changed files with 221 additions and 23 deletions

View File

@ -27,8 +27,11 @@
* Purpose: The class Array2 represents bi-dimensional arrays * Purpose: The class Array2 represents bi-dimensional arrays
* of fixed size known at run time. * of fixed size known at run time.
* The ranges of indices are user defined. * The ranges of indices are user defined.
* *
* Warning: Programs clients of such class must be independant * Class allocates one 1D array storing full data (all Rows and Columns)
* and extra 1D array storing pointers to each Row.
*
* Warning: Programs clients of such class must be independent
* of the range of the first element. Then, a C++ for * of the range of the first element. Then, a C++ for
* loop must be written like this * loop must be written like this
* *
@ -85,6 +88,20 @@ public:
public: public:
// ---------- PUBLIC METHODS ------------ // ---------- PUBLIC METHODS ------------
//! Empty constructor; should be used with caution.
//! @sa methods Resize() and Move().
NCollection_Array2()
: myLowerRow (1),
myUpperRow (0),
myLowerCol (1),
myUpperCol (0),
myData (NULL),
myStart (NULL),
myDeletable(false)
{
//
}
//! Constructor //! Constructor
NCollection_Array2(const Standard_Integer theRowLower, NCollection_Array2(const Standard_Integer theRowLower,
const Standard_Integer theRowUpper, const Standard_Integer theRowUpper,
@ -109,6 +126,23 @@ public:
*this = theOther; *this = theOther;
} }
#ifndef OCCT_NO_RVALUE_REFERENCE
//! Move constructor
NCollection_Array2 (NCollection_Array2&& theOther)
: myLowerRow (theOther.myLowerRow),
myUpperRow (theOther.myUpperRow),
myLowerCol (theOther.myLowerRow),
myUpperCol (theOther.myUpperCol),
myData (theOther.myData),
myStart (theOther.myStart),
myDeletable(theOther.myDeletable)
{
theOther.myStart = NULL;
theOther.myData = NULL;
theOther.myDeletable = false;
}
#endif
//! C array-based constructor //! C array-based constructor
NCollection_Array2(const TheItemType& theBegin, NCollection_Array2(const TheItemType& theBegin,
const Standard_Integer theRowLower, const Standard_Integer theRowLower,
@ -138,15 +172,19 @@ public:
{ return Length(); } { return Length(); }
//! Length (number of items) //! Length (number of items)
Standard_Integer Length (void) const Standard_Integer Length (void) const
{ return RowLength() * ColLength(); } { return NbRows() * NbColumns(); }
//! Returns number of rows
Standard_Integer NbRows() const { return myUpperRow - myLowerRow + 1; }
//! Returns number of columns
Standard_Integer NbColumns() const { return myUpperCol - myLowerCol + 1; }
//! Returns length of the row, i.e. number of columns //! Returns length of the row, i.e. number of columns
Standard_Integer RowLength (void) const Standard_Integer RowLength() const { return NbColumns(); }
{ return (myUpperCol-myLowerCol+1); }
//! Returns length of the column, i.e. number of rows //! Returns length of the column, i.e. number of rows
Standard_Integer ColLength (void) const Standard_Integer ColLength() const { return NbRows(); }
{ return (myUpperRow-myLowerRow+1); }
//! LowerRow //! LowerRow
Standard_Integer LowerRow (void) const Standard_Integer LowerRow (void) const
@ -179,12 +217,53 @@ public:
return *this; return *this;
} }
//! Move assignment.
//! This array will borrow all the data from theOther.
//! The moved object will be left unitialized and should not be used anymore.
NCollection_Array2& Move (NCollection_Array2& theOther)
{
if (&theOther == this)
{
return *this;
}
if (myDeletable)
{
delete[] myStart;
}
if (myData != NULL)
{
delete[] &(myData[myLowerRow]);
}
myLowerRow = theOther.myLowerRow;
myUpperRow = theOther.myUpperRow;
myLowerCol = theOther.myLowerRow;
myUpperCol = theOther.myUpperCol;
myData = theOther.myData;
myStart = theOther.myStart;
myDeletable = theOther.myDeletable;
theOther.myStart = NULL;
theOther.myData = NULL;
theOther.myDeletable = Standard_False;
return *this;
}
//! Assignment operator //! Assignment operator
NCollection_Array2& operator= (const NCollection_Array2& theOther) NCollection_Array2& operator= (const NCollection_Array2& theOther)
{ {
return Assign (theOther); return Assign (theOther);
} }
#ifndef OCCT_NO_RVALUE_REFERENCE
//! Move assignment operator; @sa Move()
NCollection_Array2& operator= (NCollection_Array2&& theOther)
{
return Move (theOther);
}
#endif
//! Constant value access //! Constant value access
const TheItemType& Value (const Standard_Integer theRow, const TheItemType& Value (const Standard_Integer theRow,
const Standard_Integer theCol) const const Standard_Integer theCol) const
@ -222,12 +301,86 @@ public:
theCol < myLowerCol || theCol > myUpperCol, "NCollection_Array2::SetValue"); theCol < myLowerCol || theCol > myUpperCol, "NCollection_Array2::SetValue");
myData[theRow][theCol] = theItem; myData[theRow][theCol] = theItem;
} }
//! Resizes the array to specified bounds.
//! No re-allocation will be done if length of array does not change,
//! but existing values will not be discarded if theToCopyData set to FALSE.
//! @param theRowLower new lower Row of array
//! @param theRowUpper new upper Row of array
//! @param theColLower new lower Column of array
//! @param theColUpper new upper Column of array
//! @param theToCopyData flag to copy existing data into new array
void Resize (Standard_Integer theRowLower,
Standard_Integer theRowUpper,
Standard_Integer theColLower,
Standard_Integer theColUpper,
Standard_Boolean theToCopyData)
{
Standard_RangeError_Raise_if (theRowUpper < theRowLower
|| theColUpper < theColLower, "NCollection_Array2::Resize");
const Standard_Integer anOldNbRows = NbRows();
const Standard_Integer anOldNbCols = NbColumns();
const Standard_Integer aLowerRowOld = myLowerRow;
const Standard_Integer aLowerColOld = myLowerCol;
const Standard_Integer aNewNbRows = theRowUpper - theRowLower + 1;
const Standard_Integer aNewNbCols = theColUpper - theColLower + 1;
TheItemType* aStartOld = myStart;
TheItemType** aTableOld = myData != NULL ? myData + aLowerRowOld : NULL;
myLowerRow = theRowLower;
myUpperRow = theRowUpper;
myLowerCol = theColLower;
myUpperCol = theColUpper;
if (aNewNbRows == anOldNbRows
&& aNewNbCols == anOldNbCols)
{
if (myLowerCol != aLowerColOld)
{
fillIndexTable (aTableOld);
}
myData = aTableOld - myLowerRow;
return;
}
if (myDeletable
&& !theToCopyData)
{
delete[] aStartOld;
}
delete[] aTableOld;
Allocate();
if (!theToCopyData)
{
myDeletable = Standard_True;
return;
}
const Standard_Integer aNbRowsToCopy = Min (anOldNbRows, aNewNbRows);
const Standard_Integer aNbColsToCopy = Min (anOldNbCols, aNewNbCols);
for (Standard_Integer aRowIter = 0; aRowIter < aNbRowsToCopy; ++aRowIter)
{
for (Standard_Integer aColIter = 0; aColIter < aNbColsToCopy; ++aColIter)
{
myStart[size_t(aRowIter) * size_t(aNewNbCols) + size_t(aColIter)] = aStartOld[size_t(aRowIter) * size_t(anOldNbCols) + size_t(aColIter)];
}
}
if (myDeletable)
{
delete[] aStartOld;
}
myDeletable = Standard_True;
}
//! Destructor - releases the memory //! Destructor - releases the memory
~NCollection_Array2 (void) ~NCollection_Array2 (void)
{ {
if (myDeletable) delete [] myStart; if (myDeletable) delete [] myStart;
delete [] &(myData[myLowerRow]); if (myData != NULL)
{
delete[] &(myData[myLowerRow]);
}
} }
private: private:
@ -236,28 +389,37 @@ public:
//! Allocate memory for the array, set up indirection table //! Allocate memory for the array, set up indirection table
void Allocate (void) void Allocate (void)
{ {
const Standard_Integer iRowSize = myUpperCol - myLowerCol + 1; const Standard_Integer aNbRows = NbRows();
const Standard_Integer iColSize = myUpperRow - myLowerRow + 1; const Standard_Integer aNbCols = NbColumns();
Standard_RangeError_Raise_if (iRowSize <= 0 || iColSize <= 0, "NCollection_Array2::Allocate"); Standard_RangeError_Raise_if (aNbRows <= 0 || aNbCols <= 0, "NCollection_Array2::Allocate");
if (myDeletable) { if (myDeletable)
{
// allocation of the data in the array // allocation of the data in the array
myStart = new TheItemType[iRowSize * iColSize]; myStart = new TheItemType[size_t(aNbRows) * size_t(aNbCols)];
Standard_OutOfMemory_Raise_if (!myStart, "NCollection_Array2 : Allocation failed"); Standard_OutOfMemory_Raise_if (!myStart, "NCollection_Array2 : Allocation failed");
} }
// else myStart is set to the beginning of the given array // else myStart is set to the beginning of the given array
TheItemType** pTable = new TheItemType* [iColSize];
Standard_OutOfMemory_Raise_if (!pTable, "NCollection_Array2 : Allocation failed");
// Items of pTable point to the '0'th items in the rows of the array TheItemType** pTable = new TheItemType* [aNbRows];
TheItemType* pRow = myStart - myLowerCol; Standard_OutOfMemory_Raise_if (!pTable, "NCollection_Array2 : Allocation failed");
for (Standard_Integer i = 0; i < iColSize; i++) fillIndexTable (pTable);
}
//! Fill index table for accessing array elements.
void fillIndexTable (TheItemType** theTable)
{
// Items of table point to the 0th items in the rows of the array
TheItemType* aRow = myStart - myLowerCol;
const Standard_Integer aNbRows = NbRows();
const Standard_Size aNbCols = NbColumns();
for (Standard_Integer aRowIter = 0; aRowIter < aNbRows; ++aRowIter)
{ {
pTable[i] = pRow; theTable[aRowIter] = aRow;
pRow += iRowSize; aRow += aNbCols;
} }
// Set myData to the '0'th row pointer of the pTable // Set myData to the 0th row pointer of the table
myData = pTable - myLowerRow; myData = theTable - myLowerRow;
} }
protected: protected:

View File

@ -716,6 +716,42 @@ static Standard_Integer QANColTestArray2(Draw_Interpretor& di, Standard_Integer
} }
QANCollection_Array2Func anArr2(LowerRow, UpperRow, LowerCol, UpperCol); QANCollection_Array2Func anArr2(LowerRow, UpperRow, LowerCol, UpperCol);
TestArray2(anArr2); TestArray2(anArr2);
// check resize
for (int aPass = 0; aPass <= 5; ++aPass)
{
Standard_Integer aNewLowerRow = LowerRow, aNewUpperRow = UpperRow, aNewLowerCol = LowerCol, aNewUpperCol = UpperCol;
switch (aPass)
{
case 0: aNewLowerRow -= 1; break;
case 1: aNewLowerCol -= 1; break;
case 2: aNewLowerRow -= 1; aNewLowerCol -= 1; break;
case 3: aNewUpperRow += 1; break;
case 4: aNewUpperCol += 1; break;
case 5: aNewUpperRow += 1; aNewUpperCol += 1; break;
}
QANCollection_Array2Func anArr2Copy = anArr2;
anArr2Copy.Resize (aNewLowerRow, aNewUpperRow, aNewLowerCol, aNewUpperCol, true);
const Standard_Integer aNbRowsMin = Min (anArr2.NbRows(), anArr2Copy.NbRows());
const Standard_Integer aNbColsMin = Min (anArr2.NbColumns(), anArr2Copy.NbColumns());
for (Standard_Integer aRowIter = 0; aRowIter < aNbRowsMin; ++aRowIter)
{
for (Standard_Integer aColIter = 0; aColIter < aNbColsMin; ++aColIter)
{
const gp_Pnt& aPnt1 = anArr2 .Value (aRowIter + anArr2.LowerRow(), aColIter + anArr2.LowerCol());
const gp_Pnt& aPnt2 = anArr2Copy.Value (aRowIter + anArr2Copy.LowerRow(), aColIter + anArr2Copy.LowerCol());
if (!aPnt1.IsEqual (aPnt2, gp::Resolution()))
{
std::cerr << "Error: 2D array is not properly resized\n";
return 1;
}
}
}
}
QANCollection_Array2Func anArr2Copy2 = anArr2;
anArr2Copy2.Resize (LowerRow - 1, UpperRow - 1, LowerCol + 1, UpperCol + 1, false);
return 0; return 0;
} }