#ifndef MARISA_VECTOR_INLINE_H_ #define MARISA_VECTOR_INLINE_H_ namespace marisa { template <typename T> Vector<T>::Vector() : buf_(NULL), objs_(NULL), size_(0), capacity_(0), fixed_(false) {} template <typename T> Vector<T>::~Vector() { if (buf_ != NULL) { for (std::size_t i = 0; i < size_; ++i) { buf_[i].~T(); } delete [] reinterpret_cast<char *>(buf_); } } template <typename T> void Vector<T>::mmap(Mapper *mapper, const char *filename, long offset, int whence) { MARISA_THROW_IF(mapper == NULL, MARISA_PARAM_ERROR); Mapper temp_mapper; temp_mapper.open(filename, offset, whence); map(temp_mapper); temp_mapper.swap(mapper); } template <typename T> void Vector<T>::map(const void *ptr, std::size_t size) { Mapper mapper(ptr, size); map(mapper); } template <typename T> void Vector<T>::map(Mapper &mapper) { UInt32 size; mapper.map(&size); Vector temp; mapper.map(&temp.objs_, size); temp.size_ = size; temp.fix(); temp.swap(this); } template <typename T> void Vector<T>::load(const char *filename, long offset, int whence) { Reader reader; reader.open(filename, offset, whence); read(reader); } template <typename T> void Vector<T>::fread(std::FILE *file) { Reader reader(file); read(reader); } template <typename T> void Vector<T>::read(int fd) { Reader reader(fd); read(reader); } template <typename T> void Vector<T>::read(std::istream &stream) { Reader reader(&stream); read(reader); } template <typename T> void Vector<T>::read(Reader &reader) { UInt32 size; reader.read(&size); Vector temp; temp.resize(size); reader.read(temp.buf_, size); temp.swap(this); } template <typename T> void Vector<T>::save(const char *filename, bool trunc_flag, long offset, int whence) const { Writer writer; writer.open(filename, trunc_flag, offset, whence); write(writer); } template <typename T> void Vector<T>::fwrite(std::FILE *file) const { Writer writer(file); write(writer); } template <typename T> void Vector<T>::write(int fd) const { Writer writer(fd); write(writer); } template <typename T> void Vector<T>::write(std::ostream &stream) const { Writer writer(&stream); write(writer); } template <typename T> void Vector<T>::write(Writer &writer) const { writer.write(size_); writer.write(objs_, size_); } template <typename T> void Vector<T>::push_back(const T &x) { MARISA_THROW_IF(fixed_, MARISA_STATE_ERROR); MARISA_THROW_IF(size_ == max_size(), MARISA_SIZE_ERROR); reserve(size_ + 1); new (&buf_[size_++]) T(x); } template <typename T> void Vector<T>::pop_back() { MARISA_THROW_IF(fixed_ || (size_ == 0), MARISA_STATE_ERROR); buf_[--size_].~T(); } template <typename T> void Vector<T>::resize(std::size_t size) { MARISA_THROW_IF(fixed_, MARISA_STATE_ERROR); reserve(size); for (std::size_t i = size_; i < size; ++i) { new (&buf_[i]) T; } for (std::size_t i = size; i < size_; ++i) { buf_[i].~T(); } size_ = (UInt32)size; } template <typename T> void Vector<T>::resize(std::size_t size, const T &x) { MARISA_THROW_IF(fixed_, MARISA_STATE_ERROR); reserve(size); for (std::size_t i = size_; i < size; ++i) { new (&buf_[i]) T(x); } for (std::size_t i = size; i < size_; ++i) { buf_[i].~T(); } size_ = (UInt32)size; } template <typename T> void Vector<T>::reserve(std::size_t capacity) { MARISA_THROW_IF(fixed_, MARISA_STATE_ERROR); MARISA_THROW_IF(capacity > max_size(), MARISA_SIZE_ERROR); if (capacity <= capacity_) { return; } std::size_t new_capacity = capacity; if (capacity_ > (capacity / 2)) { if (capacity_ > (max_size() / 2)) { new_capacity = max_size(); } else { new_capacity = capacity_ * 2; } } realloc(new_capacity); } template <typename T> void Vector<T>::shrink() { MARISA_THROW_IF(fixed_, MARISA_STATE_ERROR); if (size_ != capacity_) { realloc(size_); } } template <typename T> void Vector<T>::fix() { MARISA_THROW_IF(fixed_, MARISA_STATE_ERROR); fixed_ = true; } template <typename T> void Vector<T>::swap(Vector *rhs) { MARISA_THROW_IF(rhs == NULL, MARISA_PARAM_ERROR); Swap(&buf_, &rhs->buf_); Swap(&objs_, &rhs->objs_); Swap(&size_, &rhs->size_); Swap(&capacity_, &rhs->capacity_); Swap(&fixed_, &rhs->fixed_); } template <typename T> void Vector<T>::realloc(std::size_t new_capacity) { MARISA_THROW_IF(new_capacity > (MARISA_SIZE_MAX / sizeof(T)), MARISA_SIZE_ERROR); T * const new_buf = reinterpret_cast<T *>( new (std::nothrow) char[sizeof(T) * new_capacity]); MARISA_THROW_IF(new_buf == NULL, MARISA_MEMORY_ERROR); for (std::size_t i = 0; i < size_; ++i) { new (&new_buf[i]) T(buf_[i]); buf_[i].~T(); } delete [] reinterpret_cast<char *>(buf_); buf_ = new_buf; objs_ = new_buf; capacity_ = (UInt32)new_capacity; } } // namespace marisa #endif // MARISA_VECTOR_INLINE_H_