LUMIERA.clone/tests/library/scoped-ptrvect-test.cpp
Ichthyostega 2bec3ccd4e clean-up: make ScopedPtrVect move-assignable to use it in STL containers
This was a pre-C++11 implementation, and at that time,
I developed the ScopedHolder to allow handling non-copyable objects in STL containers

Meanwhile we have move semantics to achieve the same goal;
and since `ScopedPtrVect` shall be retained, it should be upgraded,
using the copy-and-swap approach
2025-05-31 19:10:19 +02:00

229 lines
6.4 KiB
C++
Raw Permalink Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
ScopedPtrVect(Test) - holding and owning a collection of noncopyable objects
Copyright (C)
2008, Hermann Vosseler <Ichthyostega@web.de>
  **Lumiera** is free software; you can redistribute it and/or modify it
  under the terms of the GNU General Public License as published by the
  Free Software Foundation; either version 2 of the License, or (at your
  option) any later version. See the file COPYING for further details.
* *****************************************************************/
/** @file scoped-ptrvect-test.cpp
** unit test \ref ScopedPtrVect_test
*/
#include "lib/test/run.hpp"
#include "lib/test/test-helper.hpp"
#include "lib/util.hpp"
#include "lib/scoped-ptrvect.hpp"
#include "lib/test/tracking-dummy.hpp"
namespace lib {
namespace test{
using ::Test;
using util::isnil;
using lumiera::error::LUMIERA_ERROR_ITER_EXHAUST;
typedef ScopedPtrVect<Dummy> VectD;
/****************************************************************//**
* @test ScopedPtrVect manages the lifecycle of a number of objects.
* The API is similar to a vector and allows for element access
* and iteration. Individual elements can be detached and thus
* removed from the responsibility of the container.
*/
class ScopedPtrVect_test : public Test
{
virtual void
run (Arg)
{
simpleUsage();
iterating();
detaching();
moving();
}
void
simpleUsage()
{
CHECK (0 == Dummy::checksum());
{
VectD holder;
CHECK (isnil (holder));
CHECK (0 == Dummy::checksum());
Dummy* ptr = new Dummy();
Dummy& ref = holder.manage (ptr);
CHECK (!isnil (holder));
CHECK (0 != Dummy::checksum());
CHECK (&ref==ptr);
holder.manage (new Dummy);
holder.manage (new Dummy);
CHECK (3 == holder.size());
holder.clear();
CHECK (0 == Dummy::checksum());
CHECK (isnil (holder));
holder.manage (new Dummy);
holder.manage (new Dummy);
holder.manage (new Dummy);
holder.manage (new Dummy);
holder.manage (new Dummy);
holder.manage (new Dummy);
holder.manage (new Dummy);
holder.manage (new Dummy);
holder.manage (new Dummy);
CHECK (9 == holder.size());
CHECK (0 < Dummy::checksum());
}
CHECK (0 == Dummy::checksum());
}
void
iterating()
{
CHECK (0 == Dummy::checksum());
{
VectD holder;
for (int i=0; i<16; ++i)
holder.manage(new Dummy(i));
int check=0;
VectD::iterator ii = holder.begin();
while (ii)
{
CHECK (check == ii->getVal());
++check;
++ii;
}
// Test the const iterator
check = 0;
VectD::const_iterator cii = holder.begin();
while (cii)
{
CHECK (check == cii->getVal());
++check;
++cii;
}
// Verify correct behaviour of iteration end
CHECK (not holder.end());
CHECK (isnil (holder.end()));
VERIFY_ERROR (ITER_EXHAUST, *holder.end() );
VERIFY_ERROR (ITER_EXHAUST, ++holder.end() );
CHECK (ii == holder.end());
CHECK (cii == holder.end());
VERIFY_ERROR (ITER_EXHAUST, ++ii );
VERIFY_ERROR (ITER_EXHAUST, ++cii );
}
CHECK (0 == Dummy::checksum());
}
void
detaching()
{
int id2, id3;
Dummy* extracted{nullptr};
CHECK (0 == Dummy::checksum());
{
VectD holder;
CHECK (0 == Dummy::checksum());
CHECK (isnil (holder));
holder.manage (new Dummy);
holder.manage (new Dummy);
holder.manage (new Dummy);
holder.manage (new Dummy);
holder.manage (new Dummy);
CHECK (5 == holder.size());
CHECK (0 < Dummy::checksum());
id2 = holder[2].getVal();
id3 = holder[3].getVal();
extracted = holder.detach(& holder[2]);
CHECK (id2 == extracted->getVal());
CHECK (id3 == holder[2].getVal());
CHECK (4 == holder.size());
}
CHECK (0 < Dummy::checksum()); // not all dummies are dead
CHECK (id2 == Dummy::checksum()); // #2 is alive!
extracted->setVal(id2+id3);
CHECK (id2+id3 == Dummy::checksum());
delete extracted;
CHECK (0 == Dummy::checksum());
}
void
moving()
{ {
VectD org;
VectD left;
CHECK (0 == Dummy::checksum());
org.manage (new Dummy);
org.manage (new Dummy);
org.manage (new Dummy);
CHECK (not isnil (org));
CHECK ( isnil (left));
auto sum = Dummy::checksum();
CHECK (sum > 0);
int id0 = org[0].getVal(),
id1 = org[1].getVal(),
id2 = org[2].getVal();
// create by move
VectD right{std::move (org)};
CHECK ( isnil (org));
CHECK ( isnil (left));
CHECK (not isnil (right));
CHECK (sum == Dummy::checksum());
// move-assignment
left = std::move (right);
CHECK ( isnil (org));
CHECK (not isnil (left));
CHECK ( isnil (right));
CHECK (sum == Dummy::checksum());
CHECK (id0 == left[0].getVal());
CHECK (id1 == left[1].getVal());
CHECK (id2 == left[2].getVal());
}
CHECK (0 == Dummy::checksum());
}
};
LAUNCHER (ScopedPtrVect_test, "unit common");
}} // namespace lib::test