Array.h 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113
  1. /*
  2. * Copyright (c) 2020, the SerenityOS developers.
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #pragma once
  7. #include <AK/Iterator.h>
  8. #include <AK/Span.h>
  9. namespace AK {
  10. template<typename T, size_t Size>
  11. struct Array {
  12. [[nodiscard]] constexpr T const* data() const { return __data; }
  13. [[nodiscard]] constexpr T* data() { return __data; }
  14. [[nodiscard]] constexpr size_t size() const { return Size; }
  15. [[nodiscard]] constexpr Span<T const> span() const { return { __data, Size }; }
  16. [[nodiscard]] constexpr Span<T> span() { return { __data, Size }; }
  17. [[nodiscard]] constexpr T const& at(size_t index) const
  18. {
  19. VERIFY(index < size());
  20. return __data[index];
  21. }
  22. [[nodiscard]] constexpr T& at(size_t index)
  23. {
  24. VERIFY(index < size());
  25. return __data[index];
  26. }
  27. [[nodiscard]] constexpr T const& front() const { return at(0); }
  28. [[nodiscard]] constexpr T& front() { return at(0); }
  29. [[nodiscard]] constexpr T const& back() const requires(Size > 0) { return at(Size - 1); }
  30. [[nodiscard]] constexpr T& back() requires(Size > 0) { return at(Size - 1); }
  31. [[nodiscard]] constexpr bool is_empty() const { return size() == 0; }
  32. [[nodiscard]] constexpr T const& operator[](size_t index) const { return at(index); }
  33. [[nodiscard]] constexpr T& operator[](size_t index) { return at(index); }
  34. template<typename T2, size_t Size2>
  35. [[nodiscard]] constexpr bool operator==(Array<T2, Size2> const& other) const { return span() == other.span(); }
  36. using ConstIterator = SimpleIterator<Array const, T const>;
  37. using Iterator = SimpleIterator<Array, T>;
  38. [[nodiscard]] constexpr ConstIterator begin() const { return ConstIterator::begin(*this); }
  39. [[nodiscard]] constexpr Iterator begin() { return Iterator::begin(*this); }
  40. [[nodiscard]] constexpr ConstIterator end() const { return ConstIterator::end(*this); }
  41. [[nodiscard]] constexpr Iterator end() { return Iterator::end(*this); }
  42. [[nodiscard]] constexpr operator Span<T const>() const { return span(); }
  43. [[nodiscard]] constexpr operator Span<T>() { return span(); }
  44. constexpr size_t fill(T const& value)
  45. {
  46. for (size_t idx = 0; idx < Size; ++idx)
  47. __data[idx] = value;
  48. return Size;
  49. }
  50. [[nodiscard]] constexpr T max() requires(requires(T x, T y) { x < y; })
  51. {
  52. static_assert(Size > 0, "No values to max() over");
  53. T value = __data[0];
  54. for (size_t i = 1; i < Size; ++i)
  55. value = AK::max(__data[i], value);
  56. return value;
  57. }
  58. [[nodiscard]] constexpr T min() requires(requires(T x, T y) { x > y; })
  59. {
  60. static_assert(Size > 0, "No values to min() over");
  61. T value = __data[0];
  62. for (size_t i = 1; i < Size; ++i)
  63. value = AK::min(__data[i], value);
  64. return value;
  65. }
  66. T __data[Size];
  67. };
  68. template<typename T, typename... Types>
  69. Array(T, Types...) -> Array<T, sizeof...(Types) + 1>;
  70. namespace Detail {
  71. template<typename T, size_t... Is>
  72. constexpr auto integer_sequence_generate_array([[maybe_unused]] T const offset, IntegerSequence<T, Is...>) -> Array<T, sizeof...(Is)>
  73. {
  74. return { { (offset + Is)... } };
  75. }
  76. }
  77. template<typename T, T N>
  78. constexpr static auto iota_array(T const offset = {})
  79. {
  80. static_assert(N >= T {}, "Negative sizes not allowed in iota_array()");
  81. return Detail::integer_sequence_generate_array<T>(offset, MakeIntegerSequence<T, N>());
  82. }
  83. }
  84. using AK::Array;
  85. using AK::iota_array;