Darwin Neuroevolution Framework
matrix.h
1 // Copyright 2018 The Darwin Neuroevolution Framework Authors.
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 // http://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and
13 // limitations under the License.
14 
15 #pragma once
16 
17 #include "exception.h"
18 
19 #include <third_party/json/json.h>
20 using nlohmann::json;
21 
22 #include <assert.h>
23 #include <vector>
24 using namespace std;
25 
26 namespace core {
27 
30 template <class T>
31 class ArrayView {
32  public:
34  ArrayView(T* array, size_t size) : array_(array), size_(size) {}
35 
37  T& operator[](size_t index) const {
38  assert(index < size_);
39  return array_[index];
40  }
41 
43  size_t size() const { return size_; }
44 
45  private:
46  T* const array_;
47  const size_t size_;
48 };
49 
56 template <class T>
57 struct Matrix {
59  Matrix() = default;
60 
62  Matrix(size_t rows, size_t cols) : rows(rows), cols(cols) {
63  assert(rows > 0);
64  assert(cols > 0);
65  values.resize(rows * cols);
66  }
67 
69  ArrayView<T> operator[](size_t row) {
70  assert(row < rows);
71  return { &values[row * cols], cols };
72  }
73 
75  ArrayView<const T> operator[](size_t row) const {
76  assert(row < rows);
77  return { &values[row * cols], cols };
78  }
79 
81  bool empty() const { return values.empty(); }
82 
84  friend void to_json(json& json_obj, const Matrix& m) {
85  json_obj["rows"] = m.rows;
86  json_obj["cols"] = m.cols;
87  json_obj["values"] = m.values;
88  }
89 
91  friend void from_json(const json& json_obj, Matrix& m) {
92  m.rows = json_obj.at("rows");
93  m.cols = json_obj.at("cols");
94  m.values = json_obj.at("values").get<vector<T>>();
95 
96  if ((m.rows == 0) != (m.cols == 0))
97  throw core::Exception("Failed to deserialize matrix");
98 
99  if (m.rows * m.cols != m.values.size())
100  throw core::Exception("Failed to deserialize matrix");
101  }
102 
103  size_t rows = 0;
104  size_t cols = 0;
105  vector<T> values;
106 };
107 
108 } // namespace core
ArrayView(T *array, size_t size)
Constructs an array view.
Definition: matrix.h:34
T & operator[](size_t index) const
Indexed element access.
Definition: matrix.h:37
The base for exception types in the Darwin framework.
Definition: exception.h:27
Generic utilities.
Definition: exception.h:24
ArrayView< T > operator[](size_t row)
Indexed access to a row in the matrix.
Definition: matrix.h:69
STL namespace.
A dynamically sized 2D matrix.
Definition: matrix.h:57
size_t size() const
The count of elements in the array view.
Definition: matrix.h:43
Matrix(size_t rows, size_t cols)
Constructs a matrix with the specified number of rows and columns.
Definition: matrix.h:62
friend void from_json(const json &json_obj, Matrix &m)
Conversion from json.
Definition: matrix.h:91
bool empty() const
Returns true if the matrix is empty.
Definition: matrix.h:81
friend void to_json(json &json_obj, const Matrix &m)
Conversion to json.
Definition: matrix.h:84
ArrayView< const T > operator[](size_t row) const
Indexed access to a row in the matrix.
Definition: matrix.h:75
A 2d array slice (zero-based index, known size)
Definition: matrix.h:31