/*
* Copyright (C) 2014 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef LATINIME_INT_ARRAY_VIEW_H
#define LATINIME_INT_ARRAY_VIEW_H
#include <algorithm>
#include <array>
#include <cstdint>
#include <cstring>
#include <vector>
#include "defines.h"
namespace latinime {
/**
* Helper class used to provide a read-only view of a given range of integer array. This class
* does not take ownership of the underlying integer array but is designed to be a lightweight
* object that obeys value semantics.
*
* Example:
* <code>
* bool constinsX(IntArrayView view) {
* for (size_t i = 0; i < view.size(); ++i) {
* if (view[i] == 'X') {
* return true;
* }
* }
* return false;
* }
*
* const int codePointArray[] = { 'A', 'B', 'X', 'Z' };
* auto view = IntArrayView(codePointArray, NELEMS(codePointArray));
* const bool hasX = constinsX(view);
* </code>
*/
class IntArrayView {
public:
IntArrayView() : mPtr(nullptr), mSize(0) {}
IntArrayView(const int *const ptr, const size_t size)
: mPtr(ptr), mSize(size) {}
explicit IntArrayView(const std::vector<int> &vector)
: mPtr(vector.data()), mSize(vector.size()) {}
template <size_t N>
AK_FORCE_INLINE static IntArrayView fromArray(const std::array<int, N> &array) {
return IntArrayView(array.data(), array.size());
}
// Returns a view that points one int object.
AK_FORCE_INLINE static IntArrayView singleElementView(const int *const ptr) {
return IntArrayView(ptr, 1);
}
AK_FORCE_INLINE int operator[](const size_t index) const {
ASSERT(index < mSize);
return mPtr[index];
}
AK_FORCE_INLINE bool empty() const {
return size() == 0;
}
AK_FORCE_INLINE size_t size() const {
return mSize;
}
AK_FORCE_INLINE const int *data() const {
return mPtr;
}
AK_FORCE_INLINE const int *begin() const {
return mPtr;
}
AK_FORCE_INLINE const int *end() const {
return mPtr + mSize;
}
AK_FORCE_INLINE bool contains(const int value) const {
return std::find(begin(), end(), value) != end();
}
// Returns the view whose size is smaller than or equal to the given count.
AK_FORCE_INLINE const IntArrayView limit(const size_t maxSize) const {
return IntArrayView(mPtr, std::min(maxSize, mSize));
}
AK_FORCE_INLINE const IntArrayView skip(const size_t n) const {
if (mSize <= n) {
return IntArrayView();
}
return IntArrayView(mPtr + n, mSize - n);
}
template <size_t N>
void copyToArray(std::array<int, N> *const buffer, const size_t offset) const {
ASSERT(mSize + offset <= N);
memmove(buffer->data() + offset, mPtr, sizeof(int) * mSize);
}
AK_FORCE_INLINE int firstOrDefault(const int defaultValue) const {
if (empty()) {
return defaultValue;
}
return mPtr[0];
}
AK_FORCE_INLINE int lastOrDefault(const int defaultValue) const {
if (empty()) {
return defaultValue;
}
return mPtr[mSize - 1];
}
AK_FORCE_INLINE std::vector<int> toVector() const {
return std::vector<int>(begin(), end());
}
std::vector<IntArrayView> split(const int separator, const int limit = S_INT_MAX) const {
if (limit <= 0) {
return std::vector<IntArrayView>();
}
std::vector<IntArrayView> result;
if (limit == 1) {
result.emplace_back(mPtr, mSize);
return result;
}
size_t startIndex = 0;
for (size_t i = 0; i < mSize; ++i) {
if (mPtr[i] == separator) {
result.emplace_back(mPtr + startIndex, i - startIndex);
startIndex = i + 1;
if (result.size() >= static_cast<size_t>(limit - 1)) {
break;
}
}
}
result.emplace_back(mPtr + startIndex, mSize - startIndex);
return result;
}
private:
DISALLOW_ASSIGNMENT_OPERATOR(IntArrayView);
const int *const mPtr;
const size_t mSize;
};
using WordIdArrayView = IntArrayView;
using PtNodePosArrayView = IntArrayView;
using CodePointArrayView = IntArrayView;
template <size_t size>
using WordIdArray = std::array<int, size>;
} // namespace latinime
#endif // LATINIME_MEMORY_VIEW_H