Advertisement
Not a member of Pastebin yet?
Sign Up,
it unlocks many cool features!
- octopus.h
- #pragma once
- // Тут можно подключить scopedptr.h и ptrvector.h,
- // если они вам понадобятся.
- #include "ptrvector.h"
- #include "scopedptr.h"
- #include <new> // Для исключения bad_alloc
- #include <vector>
- // Щупальце
- class Tentacle {
- public:
- explicit Tentacle(int id) noexcept
- : id_(id) {
- }
- int GetId() const noexcept {
- return id_;
- }
- Tentacle* GetLinkedTentacle() const noexcept {
- return linked_tentacle_;
- }
- void LinkTo(Tentacle& tentacle) noexcept {
- linked_tentacle_ = &tentacle;
- }
- void Unlink() noexcept {
- linked_tentacle_ = nullptr;
- }
- private:
- int id_ = 0;
- Tentacle* linked_tentacle_ = nullptr;
- };
- // Осьминог
- class Octopus {
- public:
- Octopus()
- : Octopus(8) {
- }
- explicit Octopus(int num_tentacles) {
- Tentacle* t = nullptr;
- try {
- for (int i = 1; i <= num_tentacles; ++i) {
- t = new Tentacle(i); // Может выбросить исключение bad_alloc
- tentacles_.GetItems().push_back(t); // Может выбросить исключение bad_alloc
- // Обнуляем указатель на щупальце, которое уже добавили в tentacles_,
- // чтобы не удалить его в обработчике catch повторно
- t = nullptr;
- }
- }
- catch (const std::bad_alloc&) {
- // Удаляем щупальца, которые успели попасть в контейнер tentacles_
- Cleanup();
- // Удаляем щупальце, которое создали, но не добавили в tentacles_
- delete t;
- // Конструктор не смог создать осьминога с восемью щупальцами,
- // поэтому выбрасываем исключение, чтобы сообщить вызывающему коду об ошибке
- // throw без параметров внутри catch выполняет ПЕРЕВЫБРОС пойманного исключения
- throw;
- }
- }
- ~Octopus() {
- // Осьминог владеет объектами в динамической памяти (щупальца),
- // которые должны быть удалены при его разрушении.
- // Деструктор - лучшее место, чтобы прибраться за собой.
- Cleanup();
- }
- // Добавляет новое щупальце с идентификатором,
- // равным (количество_щупалец + 1):
- // 1, 2, 3, ...
- // Возвращает ссылку на добавленное щупальце
- Tentacle& AddTentacle() {
- ScopedPtr<Tentacle>tentacle(new Tentacle(GetTentacleCount() + 1));
- tentacles_.GetItems().push_back(tentacle.GetRawPtr());
- tentacle.Release();
- return *tentacles_.GetItems().back();
- }
- int GetTentacleCount() const noexcept {
- return static_cast<int>(tentacles_.GetItems().size());
- }
- const Tentacle& GetTentacle(size_t index) const {
- return *tentacles_.GetItems().at(index);
- }
- Tentacle& GetTentacle(size_t index) {
- return *tentacles_.GetItems().at(index);
- }
- private:
- void Cleanup() {
- // Удаляем щупальца осьминога из динамической памяти
- for (Tentacle* t : tentacles_.GetItems()) {
- delete t;
- }
- // Очищаем массив указателей на щупальца
- tentacles_.GetItems().clear();
- }
- // Вектор хранит указатели на щупальца. Сами объекты щупалец находятся в куче
- PtrVector<Tentacle> tentacles_;
- };
- =======================================================================================================================================
- ptrvector.h
- #pragma once
- #include <vector>
- template <typename T>
- class PtrVector {
- public:
- PtrVector() = default;
- // Создаёт вектор указателей на копии объектов из other
- PtrVector(const PtrVector& other) {
- // Резервируем место в vector-е для хранения нужного количества элементов
- // Благодаря этому при push_back не будет выбрасываться исключение
- items_.reserve(other.items_.size());
- try {
- for (auto p : other.items_) {
- // Копируем объект, если указатель на него ненулевой
- auto p_copy = p ? new T(*p) : nullptr; // new может выбросить исключение
- // Не выбросит исключение, т. к. в vector память уже зарезервирована
- items_.push_back(p_copy);
- }
- }
- catch (...) {
- // удаляем элементы в векторе и перевыбрасываем пойманное исключение
- DeleteItems();
- throw;
- }
- }
- // Деструктор удаляет объекты в куче, на которые ссылаются указатели,
- // в векторе items_
- ~PtrVector() {
- DeleteItems();
- }
- // Возвращает ссылку на вектор указателей
- std::vector<T*>& GetItems() noexcept {
- return items_;
- }
- // Возвращает константную ссылку на вектор указателей
- std::vector<T*> const& GetItems() const noexcept {
- return items_;
- }
- private:
- void DeleteItems() noexcept {
- for (auto p : items_) {
- delete p;
- }
- }
- std::vector<T*> items_;
- };
- =======================================================================================================================================
- scopedptr.h
- #pragma once
- #include <stdexcept> // содержит std::logic_error
- template <typename T>
- class ScopedPtr {
- public:
- ScopedPtr() = default;
- explicit ScopedPtr(T* raw_ptr) noexcept
- : ptr_(raw_ptr) {
- }
- // Запрещаем копирование указателя
- ScopedPtr(const ScopedPtr&) = delete;
- ~ScopedPtr() {
- delete ptr_;
- }
- T* GetRawPtr() const noexcept {
- return ptr_;
- }
- T* Release() noexcept {
- T* p = ptr_;
- ptr_ = nullptr;
- return p;
- }
- explicit operator bool() const {
- return ptr_ != nullptr;
- }
- T* operator->() const {
- using namespace std::literals;
- if (!ptr_) {
- throw std::logic_error("Scoped ptr is null"s);
- }
- return ptr_;
- }
- T& operator*() const {
- using namespace std::literals;
- if (!ptr_) {
- throw std::logic_error("Scoped ptr is null"s);
- }
- return *ptr_;
- }
- private:
- T* ptr_ = nullptr;
- };
- =======================================================================================================================================
- main.cpp
- #include "octopus.h"
- #include <cassert>
- #include <iostream>
- using namespace std;
- int main() {
- // Проверка конструирования осьминогов
- {
- // По умолчанию осьминог имеет 8 щупалец
- Octopus default_octopus;
- assert(default_octopus.GetTentacleCount() == 8);
- // Осьминог может иметь отличное от 8 количество щупалец
- Octopus quadropus(4);
- assert(quadropus.GetTentacleCount() == 4);
- // И даже вообще не иметь щупалец
- Octopus coloboque(0);
- assert(coloboque.GetTentacleCount() == 0);
- }
- // Осьминогу можно добавлять щупальца
- {
- Octopus octopus(1);
- Tentacle* t0 = &octopus.GetTentacle(0);
- Tentacle* t1 = &octopus.AddTentacle();
- assert(octopus.GetTentacleCount() == 2);
- Tentacle* t2 = &octopus.AddTentacle();
- assert(octopus.GetTentacleCount() == 3);
- // После добавления щупалец ранее созданные щупальца не меняют своих адресов
- assert(&octopus.GetTentacle(0) == t0);
- assert(&octopus.GetTentacle(1) == t1);
- assert(&octopus.GetTentacle(2) == t2);
- for (int i = 0; i < octopus.GetTentacleCount(); ++i) {
- assert(octopus.GetTentacle(i).GetId() == i + 1);
- }
- }
- // Осьминоги могут прицепляться к щупальцам друг друга
- {
- Octopus male(2);
- Octopus female(2);
- assert(male.GetTentacle(0).GetLinkedTentacle() == nullptr);
- male.GetTentacle(0).LinkTo(female.GetTentacle(1));
- assert(male.GetTentacle(0).GetLinkedTentacle() == &female.GetTentacle(1));
- male.GetTentacle(0).Unlink();
- assert(male.GetTentacle(0).GetLinkedTentacle() == nullptr);
- }
- // Копия осьминога имеет свою собственную копию щупалец, которые
- // копируют состояние щупалец оригинального осьминога
- {
- // Перебираем осьминогов с разным количеством щупалец
- for (int num_tentacles = 0; num_tentacles < 10; ++num_tentacles) {
- Octopus male(num_tentacles);
- Octopus female(num_tentacles);
- // Пусть они хватают друг друга за щупальца
- for (int i = 0; i < num_tentacles; ++i) {
- male.GetTentacle(i).LinkTo(female.GetTentacle(num_tentacles - 1 - i));
- }
- Octopus male_copy(male);
- // Проверяем состояние щупалец копии
- assert(male_copy.GetTentacleCount() == male.GetTentacleCount());
- for (int i = 0; i < male_copy.GetTentacleCount(); ++i) {
- // Каждое щупальце копии размещается по адресу, отличному от адреса оригинального щупальца
- assert(&male_copy.GetTentacle(i) != &male.GetTentacle(i));
- // Каждое щупальце копии прицепляется к тому же щупальцу, что и оригинальное
- assert(male_copy.GetTentacle(i).GetLinkedTentacle() == male.GetTentacle(i).GetLinkedTentacle());
- }
- }
- // Если вы видите эту надпись, то разрушение осьминогов, скорее всего,
- // прошло без неопределённого поведения
- cout << "Everything is OK"s << endl;
- }
- }
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement