ITK 6.0.0
Insight Toolkit
 
Loading...
Searching...
No Matches
itkVariableLengthVector.h
Go to the documentation of this file.
1/*=========================================================================
2 *
3 * Copyright NumFOCUS
4 *
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
8 *
9 * https://www.apache.org/licenses/LICENSE-2.0.txt
10 *
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
16 *
17 *=========================================================================*/
18#ifndef itkVariableLengthVector_h
19#define itkVariableLengthVector_h
20
21#include <cassert>
22#include <algorithm>
23#include <type_traits>
24#include "itkNumericTraits.h"
26#include "itkIsNumber.h"
27#include "itkPromoteType.h"
29
30namespace itk
31{
32
33template <typename TExpr1, typename TExpr2, typename TBinaryOp>
34struct VariableLengthVectorExpression;
35
90template <typename TValue>
91class ITK_TEMPLATE_EXPORT VariableLengthVector
92{
93public:
98
110 struct AllocateRootPolicy
111 {};
112
126 struct AlwaysReallocate : AllocateRootPolicy
127 {
128 bool
129 operator()(unsigned int itkNotUsed(newSize), unsigned int itkNotUsed(oldSize)) const
130 {
131 return true;
132 }
133 };
134
152 struct NeverReallocate : AllocateRootPolicy
153 {
154 bool
155 operator()([[maybe_unused]] unsigned int newSize, [[maybe_unused]] unsigned int oldSize) const
156 {
157 itkAssertInDebugAndIgnoreInReleaseMacro(newSize == oldSize &&
158 "SetSize is expected to never change the VariableLengthVector size...");
159 return true;
160 }
161 };
162
180 struct ShrinkToFit : AllocateRootPolicy
181 {
182 bool
183 operator()(unsigned int newSize, unsigned int oldSize) const
184 {
185 return newSize != oldSize;
186 }
187 };
188
215 struct DontShrinkToFit : AllocateRootPolicy
216 {
217 bool
218 operator()(unsigned int newSize, unsigned int oldSize) const
219 {
220 return newSize > oldSize;
221 }
222 };
223
242 struct KeepValuesRootPolicy
243 {};
244
266 struct KeepOldValues : KeepValuesRootPolicy
267 {
268 template <typename TValue2>
269 void
270 operator()(unsigned int newSize, unsigned int oldSize, TValue2 * oldBuffer, TValue2 * newBuffer) const
271 {
272 itkAssertInDebugAndIgnoreInReleaseMacro(newBuffer);
273 const size_t nb = std::min(newSize, oldSize);
274 itkAssertInDebugAndIgnoreInReleaseMacro(nb == 0 || (nb > 0 && oldBuffer != nullptr));
275 std::copy_n(oldBuffer, nb, newBuffer);
276 }
277 };
278
295 struct DumpOldValues : KeepValuesRootPolicy
296 {
297 template <typename TValue2>
298 void
299 operator()(unsigned int itkNotUsed(newSize),
300 unsigned int itkNotUsed(oldSize),
301 TValue2 * itkNotUsed(oldBuffer),
302 TValue2 * itkNotUsed(newBuffer)) const
303 {}
304 };
306
307
309 using ValueType = TValue;
310 using ComponentType = TValue;
311 using RealValueType = typename NumericTraits<ValueType>::RealType;
312 using Self = VariableLengthVector;
313
315 using ElementIdentifier = unsigned int;
316
323 VariableLengthVector() = default;
324
333 explicit VariableLengthVector(unsigned int length);
334
348 VariableLengthVector(ValueType * datain, unsigned int sz, bool LetArrayManageMemory = false);
349
369 VariableLengthVector(const ValueType * datain, unsigned int sz, bool LetArrayManageMemory = false);
370
390 template <typename T>
391 VariableLengthVector(const VariableLengthVector<T> & v)
392 {
393 m_NumElements = v.Size();
394 m_LetArrayManageMemory = true;
395 if (m_NumElements != 0)
396 {
397 m_Data = this->AllocateElements(m_NumElements);
398 itkAssertInDebugAndIgnoreInReleaseMacro(m_Data != nullptr);
399 for (ElementIdentifier i = 0; i < m_NumElements; ++i)
400 {
401 this->m_Data[i] = static_cast<ValueType>(v[i]);
402 }
403 }
404 else
405 {
406 m_Data = nullptr;
407 }
408 }
409
418 VariableLengthVector(const VariableLengthVector<TValue> & v);
419
428 void
429 Swap(Self & v) noexcept
430 {
431 itkAssertInDebugAndIgnoreInReleaseMacro(m_LetArrayManageMemory == v.m_LetArrayManageMemory);
432 using std::swap;
433 swap(v.m_Data, m_Data);
434 swap(v.m_NumElements, m_NumElements);
435 }
436
444 VariableLengthVector(Self && v) noexcept;
445
454 Self &
455 operator=(Self && v) noexcept;
456
473 template <typename TExpr1, typename TExpr2, typename TBinaryOp>
474 VariableLengthVector(const VariableLengthVectorExpression<TExpr1, TExpr2, TBinaryOp> & rhs);
475
495 template <typename TExpr1, typename TExpr2, typename TBinaryOp>
496 Self &
497 operator=(const VariableLengthVectorExpression<TExpr1, TExpr2, TBinaryOp> & rhs);
498
502 void
503 Fill(const TValue & v);
504
517 template <typename T>
518 Self &
519 operator=(const VariableLengthVector<T> & v)
520 {
521 // No self assignment test is done. Indeed:
522 // - the operator already resists self assignment through a strong exception
523 // guarantee
524 // - the test becomes a pessimization as we never write
525 // VLV<const TValue> vcref(v.GetDataPointer(), v.GetSize());
526 // ...;
527 // v = vcref;
528 const ElementIdentifier N = v.Size();
529 this->SetSize(N, DontShrinkToFit(), DumpOldValues());
530 for (ElementIdentifier i = 0; i < N; ++i)
531 {
532 this->m_Data[i] = static_cast<ValueType>(v[i]);
533 }
534 return *this;
535 }
536
552 Self &
553 operator=(const Self & v);
554
563 Self &
564 FastAssign(const Self & v);
565
573 Self &
574 operator=(const TValue & v);
575
578 unsigned int
579 Size() const
580 {
581 return m_NumElements;
582 }
583 unsigned int
584 GetSize() const
585 {
586 return m_NumElements;
587 }
588 unsigned int
589 GetNumberOfElements() const
590 {
591 return m_NumElements;
592 }
595 TValue &
596 operator[](unsigned int i)
597 {
598 return this->m_Data[i];
599 }
600
602 const TValue &
603 operator[](unsigned int i) const
604 {
605 return this->m_Data[i];
606 }
607
609 const TValue &
610 GetElement(unsigned int i) const
611 {
612 return m_Data[i];
613 }
614
616 void
617 SetElement(unsigned int i, const TValue & value)
618 {
619 m_Data[i] = value;
620 }
621
655 template <typename TReallocatePolicy, typename TKeepValuesPolicy>
656 void
657 SetSize(unsigned int sz, TReallocatePolicy reallocatePolicy, TKeepValuesPolicy keepValues);
658
669 void
670 SetSize(unsigned int sz, bool destroyExistingData = true)
671 {
672 // Stays compatible with previous code version
673 // And works around the fact C++03 template functions can't have default
674 // arguments on template types.
675 if (destroyExistingData)
676 {
677 SetSize(sz, AlwaysReallocate(), KeepOldValues());
678 }
679 else
680 {
681 SetSize(sz, ShrinkToFit(), KeepOldValues());
682 }
683 }
684
687 void
688 DestroyExistingData();
689
702 void
703 SetData(TValue * datain, bool LetArrayManageMemory = false);
704
719 void
720 SetData(TValue * datain, unsigned int sz, bool LetArrayManageMemory = false);
721
732 ~VariableLengthVector();
733
748 void
749 Reserve(ElementIdentifier size);
750
755 TValue *
756 AllocateElements(ElementIdentifier size) const;
757
758 const TValue *
759 GetDataPointer() const
760 {
761 return m_Data;
762 }
763
766 Self &
767 operator--()
768 {
769 for (ElementIdentifier i = 0; i < m_NumElements; ++i)
770 {
771 this->m_Data[i] -= static_cast<ValueType>(1.0);
772 }
773 return *this;
774 }
775
777 Self &
778 operator++() // prefix operator ++v;
779 {
780 for (ElementIdentifier i = 0; i < m_NumElements; ++i)
781 {
782 this->m_Data[i] += static_cast<ValueType>(1.0);
783 }
784 return *this;
785 }
786
789 Self
790 operator--(int) // postfix operator v--;
791 {
792 Self tmp(*this);
793
794 --tmp;
795 return tmp;
796 }
797
799 Self
800 operator++(int) // postfix operator v++;
801 {
802 Self tmp(*this);
803
804 ++tmp;
805 return tmp;
806 }
807
816 template <typename T>
817 Self &
818 operator-=(const VariableLengthVector<T> & v)
819 {
820 itkAssertInDebugAndIgnoreInReleaseMacro(m_NumElements == v.GetSize());
821 for (ElementIdentifier i = 0; i < m_NumElements; ++i)
822 {
823 m_Data[i] -= static_cast<ValueType>(v[i]);
824 }
825 return *this;
826 }
827
829 Self &
830 operator-=(TValue s)
831 {
832 for (ElementIdentifier i = 0; i < m_NumElements; ++i)
833 {
834 m_Data[i] -= s;
835 }
836 return *this;
837 }
838
847 template <typename T>
848 Self &
849 operator+=(const VariableLengthVector<T> & v)
850 {
851 itkAssertInDebugAndIgnoreInReleaseMacro(m_NumElements == v.GetSize());
852 for (ElementIdentifier i = 0; i < m_NumElements; ++i)
853 {
854 m_Data[i] += static_cast<ValueType>(v[i]);
855 }
856 return *this;
857 }
858
860 Self &
861 operator+=(TValue s)
862 {
863 for (ElementIdentifier i = 0; i < m_NumElements; ++i)
864 {
865 m_Data[i] += s;
866 }
867 return *this;
868 }
869
879 template <typename TExpr1, typename TExpr2, typename TBinaryOp>
880 Self &
881 operator+=(const VariableLengthVectorExpression<TExpr1, TExpr2, TBinaryOp> & rhs)
882 {
883 itkAssertInDebugAndIgnoreInReleaseMacro(rhs.Size() == Size());
884 for (ElementIdentifier i = 0; i < m_NumElements; ++i)
885 {
886 m_Data[i] += static_cast<ValueType>(rhs[i]);
887 }
888 return *this;
889 }
890
900 template <typename TExpr1, typename TExpr2, typename TBinaryOp>
901 Self &
902 operator-=(const VariableLengthVectorExpression<TExpr1, TExpr2, TBinaryOp> & rhs)
903 {
904 itkAssertInDebugAndIgnoreInReleaseMacro(rhs.Size() == Size());
905 for (ElementIdentifier i = 0; i < m_NumElements; ++i)
906 {
907 m_Data[i] -= static_cast<ValueType>(rhs[i]);
908 }
909 return *this;
910 }
911
917 template <typename T>
918 Self &
919 operator*=(T s)
920 {
921 const ValueType & sc = static_cast<ValueType>(s);
922 for (ElementIdentifier i = 0; i < m_NumElements; ++i)
923 {
924 m_Data[i] *= sc;
925 }
926 return *this;
927 }
928
932 Self &
933 operator*=(TValue s)
934 {
935 for (ElementIdentifier i = 0; i < m_NumElements; ++i)
936 {
937 m_Data[i] *= s;
938 }
939 return *this;
940 }
941
948 template <typename T>
949 Self &
950 operator/=(T s)
951 {
952 const RealValueType sc = s;
953 for (ElementIdentifier i = 0; i < m_NumElements; ++i)
954 {
955 m_Data[i] = static_cast<ValueType>(static_cast<RealValueType>(m_Data[i]) / sc);
956 }
957 return *this;
958 }
959
964 Self &
965 operator-(); // negation operator
966
967 bool
968 operator==(const Self & v) const;
969
970 ITK_UNEQUAL_OPERATOR_MEMBER_FUNCTION(Self);
971
973 RealValueType
974 GetNorm() const;
975
977 RealValueType
978 GetSquaredNorm() const;
979
981 bool
982 IsAProxy() const
983 {
984 return !m_LetArrayManageMemory;
985 }
986
987private:
988 bool m_LetArrayManageMemory{ true }; // if true, the array is responsible
989 // for memory of data
990 TValue * m_Data{}; // Array to hold data
991 ElementIdentifier m_NumElements{ 0 };
992};
993
995namespace mpl
996{
1006template <typename T>
1007struct IsArray : FalseType
1008{};
1009
1011template <typename T>
1012struct IsArray<itk::VariableLengthVector<T>> : TrueType
1013{};
1014
1015template <typename TExpr1, typename TExpr2, typename TBinaryOp>
1016struct IsArray<VariableLengthVectorExpression<TExpr1, TExpr2, TBinaryOp>> : TrueType
1017{};
1019} // namespace mpl
1021
1022namespace Details
1023{
1025
1037template <typename TExpr>
1038struct GetType
1039{
1040 using Type = TExpr;
1041
1045 static Type
1046 Load(const Type & v, unsigned int itkNotUsed(idx))
1047 {
1048 return v;
1049 }
1050};
1051
1062template <typename TExpr1, typename TExpr2>
1063inline std::enable_if_t<mpl::And<mpl::IsArray<TExpr1>, mpl::IsArray<TExpr2>>::Value, unsigned int>
1064GetSize(const TExpr1 & lhs, [[maybe_unused]] const TExpr2 & rhs)
1065{
1066 itkAssertInDebugAndIgnoreInReleaseMacro(lhs.Size() == rhs.Size());
1067 return lhs.Size();
1068}
1069
1071
1080template <typename TExpr1, typename TExpr2>
1081inline std::enable_if_t<mpl::And<mpl::IsArray<TExpr1>, mpl::Not<mpl::IsArray<TExpr2>>>::Value, unsigned int>
1082GetSize(const TExpr1 & lhs, const TExpr2 & itkNotUsed(rhs))
1083{
1084 return lhs.Size();
1085}
1086
1096template <typename TExpr1, typename TExpr2>
1097inline std::enable_if_t<mpl::And<mpl::IsArray<TExpr2>, mpl::Not<mpl::IsArray<TExpr1>>>::Value, unsigned int>
1098GetSize(const TExpr1 & itkNotUsed(lhs), const TExpr2 & rhs)
1099{
1100 return rhs.Size();
1101}
1102
1103template <typename T>
1104struct GetType<VariableLengthVector<T>>
1105{
1106 using Type = T;
1107 static Type
1108 Load(const VariableLengthVector<T> & v, unsigned int idx)
1109 {
1110 return v[idx];
1111 }
1112};
1113template <typename TExpr1, typename TExpr2, typename TBinaryOp>
1114struct GetType<VariableLengthVectorExpression<TExpr1, TExpr2, TBinaryOp>>
1115{
1116 using Type = typename VariableLengthVectorExpression<TExpr1, TExpr2, TBinaryOp>::ResType;
1117 static Type
1118 Load(const VariableLengthVectorExpression<TExpr1, TExpr2, TBinaryOp> & v, unsigned int idx)
1119 {
1120 return v[idx];
1121 }
1122};
1124
1125namespace op
1126{
1139template <typename TExpr1, typename TExpr2>
1140struct CanBeAddedOrSubtracted
1141 : mpl::Or<mpl::And<mpl::IsArray<TExpr1>, mpl::IsArray<TExpr2>>,
1142 mpl::And<mpl::IsArray<TExpr1>, mpl::IsNumber<TExpr2>>,
1143 mpl::And<mpl::IsNumber<TExpr1>, mpl::IsArray<TExpr2>>>
1144{};
1145
1157template <typename TExpr1, typename TExpr2>
1158struct CanBeMultiplied
1159 : mpl::Or<mpl::And<mpl::IsArray<TExpr1>, mpl::IsNumber<TExpr2>>,
1160 mpl::And<mpl::IsNumber<TExpr1>, mpl::IsArray<TExpr2>>>
1161{};
1162
1174template <typename TExpr1, typename TExpr2>
1175struct CanBeDivided : mpl::And<mpl::IsArray<TExpr1>, mpl::IsNumber<TExpr2>>
1176{};
1177
1178} // namespace op
1179} // namespace Details
1181
1206template <typename TExpr1, typename TExpr2, typename TBinaryOp>
1207struct VariableLengthVectorExpression
1208{
1209 VariableLengthVectorExpression(const TExpr1 & lhs, const TExpr2 & rhs)
1210 : m_lhs(lhs)
1211 , m_rhs(rhs)
1212 {
1213 // Not necessary actually as end-user/developer is not expected to
1214 // provide new BinaryOperations
1215 static_assert(std::is_base_of_v<Details::op::BinaryOperationConcept, TBinaryOp>,
1216 "The Binary Operation shall inherit from BinaryOperationConcept");
1217 }
1218
1220 unsigned int
1221 Size() const
1222 {
1223 return Details::GetSize(m_lhs, m_rhs);
1224 }
1225
1227 using ResType =
1228 typename mpl::PromoteType<typename Details::GetType<TExpr1>::Type, typename Details::GetType<TExpr2>::Type>::Type;
1230 using RealValueType = typename NumericTraits<ResType>::RealType;
1231
1242 ResType
1243 operator[](unsigned int idx) const
1244 {
1245 itkAssertInDebugAndIgnoreInReleaseMacro(idx < Size());
1246 return TBinaryOp::Apply(Details::GetType<TExpr1>::Load(m_lhs, idx), Details::GetType<TExpr2>::Load(m_rhs, idx));
1247 }
1248
1250 RealValueType
1251 GetNorm() const;
1252
1254 RealValueType
1255 GetSquaredNorm() const;
1256
1257private:
1258 const TExpr1 & m_lhs;
1259 const TExpr2 & m_rhs;
1260};
1261
1271template <typename TExpr1, typename TExpr2>
1272inline std::enable_if_t<Details::op::CanBeAddedOrSubtracted<TExpr1, TExpr2>::Value,
1273 VariableLengthVectorExpression<TExpr1, TExpr2, Details::op::Plus>>
1274operator+(const TExpr1 & lhs, const TExpr2 & rhs)
1275{
1276 return VariableLengthVectorExpression<TExpr1, TExpr2, Details::op::Plus>(lhs, rhs);
1277}
1278
1288template <typename TExpr1, typename TExpr2>
1289inline std::enable_if_t<Details::op::CanBeAddedOrSubtracted<TExpr1, TExpr2>::Value,
1290 VariableLengthVectorExpression<TExpr1, TExpr2, Details::op::Sub>>
1291operator-(const TExpr1 & lhs, const TExpr2 & rhs)
1292{
1293 return VariableLengthVectorExpression<TExpr1, TExpr2, Details::op::Sub>(lhs, rhs);
1294}
1295
1304template <typename TExpr1, typename TExpr2>
1305inline std::enable_if_t<Details::op::CanBeMultiplied<TExpr1, TExpr2>::Value,
1306 VariableLengthVectorExpression<TExpr1, TExpr2, Details::op::Mult>>
1307operator*(const TExpr1 & lhs, const TExpr2 & rhs)
1308{
1309 return VariableLengthVectorExpression<TExpr1, TExpr2, Details::op::Mult>(lhs, rhs);
1310}
1311
1319template <typename TExpr1, typename TExpr2>
1320inline std::enable_if_t<Details::op::CanBeDivided<TExpr1, TExpr2>::Value,
1321 VariableLengthVectorExpression<TExpr1, TExpr2, Details::op::Div>>
1322operator/(const TExpr1 & lhs, const TExpr2 & rhs)
1323{
1324 return VariableLengthVectorExpression<TExpr1, TExpr2, Details::op::Div>(lhs, rhs);
1325}
1326
1330template <typename TExpr1, typename TExpr2, typename TBinaryOp>
1331std::ostream &
1332operator<<(std::ostream & os, const VariableLengthVectorExpression<TExpr1, TExpr2, TBinaryOp> & v)
1333{
1334 os << '[';
1335 if (v.Size() != 0)
1336 {
1337 os << v[0];
1338 for (unsigned int i = 1, N = v.Size(); i != N; ++i)
1339 {
1340 os << ", " << v[i];
1341 }
1342 }
1343 return os << ']';
1344}
1345
1351template <typename TExpr>
1352inline std::enable_if_t<mpl::IsArray<TExpr>::Value, typename TExpr::RealValueType>
1353GetNorm(const TExpr & v)
1354{
1355 return static_cast<typename TExpr::RealValueType>(std::sqrt(static_cast<double>(GetSquaredNorm(v))));
1356}
1357
1363template <typename TExpr>
1364inline std::enable_if_t<mpl::IsArray<TExpr>::Value, typename TExpr::RealValueType>
1365GetSquaredNorm(const TExpr & v)
1366{
1367 using RealValueType = typename TExpr::RealValueType;
1368 RealValueType sum = 0.0;
1369 for (unsigned int i = 0, N = v.Size(); i < N; ++i)
1370 {
1371 const RealValueType value = v[i];
1372 sum += value * value;
1373 }
1374 return sum;
1375}
1376
1378
1382template <typename TValue>
1383std::ostream &
1384operator<<(std::ostream & os, const VariableLengthVector<TValue> & arr)
1385{
1386 const unsigned int length = arr.Size();
1387 const int last = static_cast<unsigned int>(length) - 1;
1388
1389 os << '[';
1390 for (int i = 0; i < last; ++i)
1391 {
1392 os << arr[i] << ", ";
1393 }
1394 if (length >= 1)
1395 {
1396 os << arr[last];
1397 }
1398 os << ']';
1399 return os;
1400}
1401
1403
1421template <typename T>
1422inline void
1424{
1425 l_.Swap(r_);
1426}
1427
1428} // namespace itk
1429
1431
1432#ifndef ITK_MANUAL_INSTANTIATION
1433# include "itkVariableLengthVector.hxx"
1434#endif
1435
1436#endif
bool ITKIOXML_EXPORT operator==(itk::FancyString &s, const std::string &)
The "itk" namespace contains all Insight Segmentation and Registration Toolkit (ITK) classes....
void swap(Array< T > &a, Array< T > &b) noexcept
Definition itkArray.h:247
ITKCommon_EXPORT std::ostream & operator<<(std::ostream &out, typename AnatomicalOrientation::CoordinateEnum value)
CovariantVector< T, VVectorDimension > operator*(const T &scalar, const CovariantVector< T, VVectorDimension > &v)
ConstNeighborhoodIterator< TImage > operator-(const ConstNeighborhoodIterator< TImage > &it, const typename ConstNeighborhoodIterator< TImage >::OffsetType &ind)
ConstNeighborhoodIterator< TImage > operator+(const ConstNeighborhoodIterator< TImage > &it, const typename ConstNeighborhoodIterator< TImage >::OffsetType &ind)