PODIO Collection as a Container
Comparison of the PODIO Collection
s with a C++ named requirement Container.
The PODIO Collection
s interface was designed to mimic the standard Container interface, in particular std::vector
. Perfect compliance with the Container is not achieved as the Collection
s are concerned with additional semantics such as mutable/immutable element access, associations and relations, and IO which that are not part of Container.
On the implementation level most of the differences with respect to the Container comes from the fact that in order to satisfy the additional semantics a Collection
doesn’t directly store user layer objects. Instead, data layer objects are stored and user layer objects are constructed and returned when needed. Similarly, the Collection
iterators operate on the user layer objects but don’t expose Collection
’s storage directly to the users. Instead, they construct and return user layer objects when needed.
In other words, a Collection
utilizes the user layer type as a reference type instead of using plain references (&
or &&
) to stored data layer types.
As a consequence some of the standard algorithms may not work with PODIO Collection
iterators. See standard algorithm documentation below.
The following tables list the compliance of a PODIO generated collection with the Container named requirement, stating which member types, interfaces, or concepts are fulfilled and which are not. Additionally, there are some comments explaining missing parts or pointing out differences in behaviour.
Container Types
Name |
Type |
Requirements |
Fulfilled by Collection? |
Comment |
---|---|---|---|---|
|
|
✔️ yes |
Defined as an immutable user layer object type |
|
|
|
❌ no |
Not defined |
|
|
|
❌ no |
Not defined |
|
|
Iterator whose |
LegacyForwardIterator convertible to |
❌ no |
Defined as podio |
|
Constant iterator whose |
❌ no |
Defined as podio |
|
|
Signed integer |
Must be the same as |
❌ no |
|
|
Unsigned integer |
Large enough to represent all positive values of |
✔️ yes |
Container member functions and operators
Expression |
Return type |
Semantics |
Fulfilled by Collection? |
Comment |
---|---|---|---|---|
|
|
Creates an empty container |
✔️ yes |
|
|
|
Creates a copy of |
❌ no |
Not defined, non-copyable by design |
|
|
Moves |
✔️ yes |
|
|
|
Destroys or copy-assigns all elements of |
❌ no |
Not defined, non-copyable by design |
|
|
Destroys or move-assigns all elements of |
✔️ yes |
|
|
|
Destroys all elements of |
✔️ yes |
Invalidates all handles retrieved from this collection |
|
|
Iterator to the first element of |
✔️ yes |
|
|
|
Iterator to one past the last element of |
✔️ yes |
|
|
|
Same as |
✔️ yes |
|
|
|
Same as |
✔️ yes |
|
|
Convertible to |
Same as |
❌ no |
Not defined |
|
Convertible to |
Same as |
❌ no |
Not defined |
|
|
Exchanges the values of |
❌ no |
Not defined |
|
|
Same as |
❌ no |
|
|
|
Same as |
✔️ yes |
|
|
|
|
✔️ yes |
|
|
Convertible to |
Same as |
✔️ yes |
Collection as an AllocatorAwareContainer
The C++ standard specifies AllocatorAwareContainer for containers that can use other allocators beside the default allocator.
PODIO collections don’t provide a customization point for allocators and use only the default allocator. Therefore they are not AllocatorAwareContainers.
AllocatorAwareContainer types
Name |
Requirements |
Fulfilled by Collection? |
Comment |
---|---|---|---|
|
|
❌ no |
|
AllocatorAwareContainer expression and statements
The PODIO Collections currently are not checked against expression and statements requirements for AllocatorAwareContainer.
Collection iterators as an Iterator
The C++ specifies a set of named requirements for iterators. Starting with C++20 the standard specifies also iterator concepts. The requirements imposed by the concepts and named requirements are similar but not identical.
In the following tables a convention from Collection
is used: iterator
stands for PODIO MutableCollectionIterator
and const_iterator
stands for PODIO CollectionIterator
.
Iterator summary
Named requirement |
|
|
---|---|---|
❌ no (see below) |
❌ no (see below) |
|
❌ no (see below) |
❌ no (see below) |
|
❌ no (see below) |
❌ no (see below) |
|
❌ no (see below) |
❌ no (see below) |
|
❌ no |
❌ no |
|
❌ no |
❌ no |
|
❌ no |
❌ no |
Concept |
|
|
---|---|---|
|
❌ no |
❌ no |
|
❌ no |
❌ no |
|
❌ no |
❌ no |
|
❌ no |
❌ no |
|
❌ no |
❌ no |
|
❌ no |
❌ no |
|
❌ no |
❌ no |
|
❌ no |
❌ no |
|
❌ no |
❌ no |
|
❌ no |
❌ no |
|
❌ no |
❌ no |
LegacyIterator
Requirement |
Fulfilled by |
Comment |
---|---|---|
❌ no / ❌ no |
Move constructor and copy constructor not defined |
|
❌ no / ❌ no |
Move assignment and copy assignment not defined |
|
✔️ yes / ✔️ yes |
||
❌ no / ❌ no |
||
|
❌ no / ❌ no |
Not defined |
|
❌ no / ❌ no |
Not defined |
|
❌ no / ❌ no |
Not defined |
|
❌ no / ❌ no |
Not defined |
|
❌ no / ❌ no |
Not defined |
Expression |
Return type |
Semantics |
Fulfilled by |
Comment |
---|---|---|---|---|
|
Unspecified |
✔️ yes / ✔️ yes |
||
|
|
✔️ yes / ✔️ yes |
LegacyInputIterator
Requirement |
Fulfilled by |
Comment |
---|---|---|
❌ no / ❌ no |
||
✔️ yes / ✔️ yes |
Expression |
Return type |
Semantics |
Fulfilled by |
Comment |
---|---|---|---|---|
|
Contextually convertible to |
Same as |
✔️ yes / ✔️ yes |
|
|
|
❌ no / ❌ no |
|
|
|
Same as |
✔️ yes / ✔️ yes |
||
|
|
✔️ yes / ✔️ yes |
||
|
Same as |
❌ no / ❌ no |
Post-increment not defined |
|
|
Convertible to |
Same as |
❌ no / ❌ no |
Post-increment and |
LegacyForwardIterator
In addition to the LegacyForwardIterator the C++ standard specifies also the mutable LegacyForwardIterator, which is both LegacyForwardIterator and LegacyOutputIterator. The term mutable used in this context doesn’t imply mutability in the sense used in the PODIO.
Requirement |
Fulfilled by |
Comment |
---|---|---|
❌ no / ❌ no |
||
❌ no / ❌ no |
Value initialization not defined |
|
If mutable iterator then |
❌ no / ❌ no |
|
❌ no / ❌ no |
Copy constructor not defined |
|
❌ no / ❌ no |
Value initialization not defined |
Expression |
Return type |
Semantics |
Fulfilled by |
Comment |
---|---|---|---|---|
|
|
Same as |
❌ no / ❌ no |
Post-increment not defined |
|
|
❌ no / ❌ no |
Post-increment and |
LegacyOutputIterator
Requirement |
Fulfilled by |
Comment |
---|---|---|
❌ no / ❌ no |
||
Is pointer type or class type |
✔️ yes / ✔️ yes |
Expression |
Return type |
Semantics |
Fulfilled by |
Comment |
---|---|---|---|---|
|
❗ attention / ❗ attention |
Defined but an assignment doesn’t modify objects inside collection |
||
|
|
✔️ yes / ✔️ yes |
||
|
Convertible to |
Same as |
❌ no / ❌ no |
Post-increment not defined |
|
Same as |
❌ no / ❌ no |
Post-increment not defined |
Collection iterators and standard iterator adaptors
Adaptor |
Compatible with Collection? |
Comment |
---|---|---|
|
❌ no |
|
|
❗ attention |
Compatible only with SubsetCollections, otherwise throws |
|
❌ no |
|
|
❌ no |
|
|
❌ no |
|
|
❌ no |
|
|
❌ no |
|
Collection and standard algorithms
Most of the standard algorithms require the iterators to be at least InputIterator. The iterators of PODIO collection don’t fulfil this requirement, therefore they are not compatible with standard algorithms according to the specification.
In practice, some algorithms may still compile with the collections depending on the implementation of a given algorithm. In general, the standard algorithms mutating a collection will give wrong results, while the standard algorithms not mutating a collection in principle should give correct results if they compile.
Standard range algorithms
The standard range algorithm use constrains to operate at least on std::input_iterator
s and std::ranges::input_range
s. The iterators of PODIO collection don’t model these concepts, therefore can’t be used with standard range algorithms. The range algorithms won’t compile with PODIO Collection
iterators.