Skip to content

Commit 416269a

Browse files
committed
Кастомный итератор по древовидной структуре.
1 parent af1cd68 commit 416269a

8 files changed

+302
-0
lines changed

README.md

+4
Original file line numberDiff line numberDiff line change
@@ -137,6 +137,10 @@
137137

138138
- [std::unordered_map при своём разрушении не вызывает деструктор для значений хранящихся по указателю](std/collections/map/unordered-map-el-dtor)
139139

140+
### Итераторы
141+
142+
- [Кастомный итератор по древовидной структуре](it/tree-iter)
143+
140144
## Потоки данных
141145

142146
- [Потоки из стандартной библиотеки](stream/streams)

it/tree-iter/.gitignore

+54
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
# ---------------------------------------------------------------------------
2+
# https://github.com/github/gitignore/blob/master/Global/Linux.gitignore
3+
4+
*~
5+
6+
# temporary files which can be created if a process still has a handle open of a deleted file
7+
.fuse_hidden*
8+
9+
# KDE directory preferences
10+
.directory
11+
12+
# Linux trash folder which might appear on any partition or disk
13+
.Trash-*
14+
15+
# .nfs files are created when an open file is removed but is still being accessed
16+
.nfs*
17+
18+
# ---------------------------------------------------------------------------
19+
# https://github.com/github/gitignore/blob/master/C++.gitignore
20+
21+
# Prerequisites
22+
*.d
23+
24+
# Compiled Object files
25+
*.slo
26+
*.lo
27+
*.o
28+
*.obj
29+
30+
# Precompiled Headers
31+
*.gch
32+
*.pch
33+
34+
# Compiled Dynamic libraries
35+
*.so
36+
*.dylib
37+
*.dll
38+
39+
# Fortran module files
40+
*.mod
41+
*.smod
42+
43+
# Compiled Static libraries
44+
*.lai
45+
*.la
46+
*.a
47+
*.lib
48+
49+
# Executables
50+
*.exe
51+
*.out
52+
*.app
53+
54+
# ---------------------------------------------------------------------------

it/tree-iter/REFERENCES.md

+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
- [итераторы - Класс-итератор c++ - Stack Overflow на русском](https://ru.stackoverflow.com/questions/717528/%D0%9A%D0%BB%D0%B0%D1%81%D1%81-%D0%B8%D1%82%D0%B5%D1%80%D0%B0%D1%82%D0%BE%D1%80-c)
2+
3+
- [std::iterator is deprecated: Why, What It Was, and What to Use Instead - Fluent C++](https://www.fluentcpp.com/2018/05/08/std-iterator-deprecated/)
4+
- [Делаем свой итератор / Хабр](https://habr.com/ru/post/265491/)
5+
- [Недооценённые итераторы / Хабр](https://habr.com/ru/post/122283/)
6+
- [C++ iterator example (and an iterable range) – Andy Balaam's Blog](https://www.artificialworlds.net/blog/2017/05/11/c-iterator-example-and-an-iterable-range/)
7+
- [Jacek's C++ Blog · Wrapping Algorithms into Iterators](https://blog.galowicz.de/2016/09/04/algorithms_in_iterators/)
8+
- [Defining a custom iterator in C++ | Lorenzo Toso](https://lorenzotoso.wordpress.com/2016/01/13/defining-a-custom-iterator-in-c/)
9+
- [Defining C++ Iterators](http://users.cs.northwestern.edu/~riesbeck/programming/c++/stl-iterator-define.html)
10+
- [Generating Sequences | Just Software Solutions - Custom Software Development](https://www.justsoftwaresolutions.co.uk/cplusplus/generating_sequences.html)
11+
12+
- [<iterator> - C++ Reference](http://www.cplusplus.com/reference/iterator/)
13+
14+
- [C++ | Итераторы](https://metanit.com/cpp/tutorial/7.3.php)
15+
- [Урок №198. Итераторы STL | Уроки С++ - Ravesli](https://ravesli.com/urok-198-iteratory-stl/)
16+
17+
- [c++ - How to implement operator-> for iterator type? - Stack Overflow](https://stackoverflow.com/questions/9007775/how-to-implement-operator-for-iterator-type)
18+
- [c++ - How to implement operator-> for an iterator that constructs its values on-demand? - Stack Overflow](https://stackoverflow.com/questions/27582845/how-to-implement-operator-for-an-iterator-that-constructs-its-values-on-demand)
19+
20+
- [перегрузка оператора -> (стрелка :) ) - C/C++ - RSDN](https://rsdn.org/forum/cpp/2705999.all)
21+
- [Overloading member access operators ->, .* (C++) - Stack Overflow](https://stackoverflow.com/questions/8777845/overloading-member-access-operators-c)
22+
23+
- [c++ - Writing a custom iterator -- what to do if you're at the end of the array? - Stack Overflow](https://stackoverflow.com/questions/2706539/writing-a-custom-iterator-what-to-do-if-youre-at-the-end-of-the-array)
24+
25+
- [std::ptrdiff_t - cppreference.com](https://en.cppreference.com/w/cpp/types/ptrdiff_t)
26+
27+
- [Implementing an iterator over binary (or arbitrary) tree using C++ 11 - Stack Overflow](https://stackoverflow.com/questions/12684191/implementing-an-iterator-over-binary-or-arbitrary-tree-using-c-11)
28+
29+
- [c++ - How to correctly implement custom iterators and const_iterators? - Stack Overflow](https://stackoverflow.com/questions/3582608/how-to-correctly-implement-custom-iterators-and-const-iterators)
30+
31+
- [Sample C++/STL custom iterator](https://gist.github.com/jeetsukumaran/307264)
32+
- [qt/qlist.h at master · radekp/qt](https://github.com/radekp/qt/blob/master/src/corelib/tools/qlist.h#L162)
33+
34+
- [Example of STL compatible custom containers + iterators : cpp](https://www.reddit.com/r/cpp/comments/9nz3xl/example_of_stl_compatible_custom_containers/)
35+
36+
- [Iterator Facade - 1.68.0](https://www.boost.org/doc/libs/1_68_0/libs/iterator/doc/iterator_facade.html)

it/tree-iter/TreeNode.cc

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
#include "TreeNode.h"
2+
#include "TreeNodeIterator.h"
3+
4+
TreeNode::TreeNode(int value) {
5+
value_ = value;
6+
}
7+
8+
int TreeNode::GetValue() const {
9+
return value_;
10+
}
11+
12+
void TreeNode::AddChildNode(TreeNode *node) {
13+
children_.push_back(node);
14+
}
15+
16+
size_t TreeNode::GetChildrenCount() const {
17+
return children_.size();
18+
}
19+
20+
TreeNode * TreeNode::GetChildAtIndex(size_t index) const {
21+
return children_.at(index);
22+
}
23+
24+
TreeNodeIterator TreeNode::begin() {
25+
return TreeNodeIterator(this);
26+
}
27+
28+
TreeNodeIterator TreeNode::end() {
29+
TreeNode *node = this;
30+
while (node->GetChildrenCount()) {
31+
node = node->GetChildAtIndex(node->GetChildrenCount() - 1);
32+
}
33+
return TreeNodeIterator(nullptr);
34+
}

it/tree-iter/TreeNode.h

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
#ifndef TreeNode_h
2+
#define TreeNode_h
3+
4+
#include <vector>
5+
6+
7+
class TreeNodeIterator;
8+
class TreeNode {
9+
std::vector<TreeNode *> children_;
10+
int value_ = -1;
11+
public:
12+
explicit TreeNode(int value);
13+
~TreeNode() = default;
14+
int GetValue() const;
15+
void AddChildNode(TreeNode *node);
16+
size_t GetChildrenCount() const;
17+
TreeNode * GetChildAtIndex(size_t index) const;
18+
TreeNodeIterator begin();
19+
TreeNodeIterator end();
20+
};
21+
22+
#endif /* TreeNode_h */

it/tree-iter/TreeNodeIterator.h

+88
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,88 @@
1+
#ifndef TreeNodeIterator_h
2+
#define TreeNodeIterator_h
3+
4+
#include <iostream>
5+
#include <stack>
6+
#include "TreeNode.h"
7+
8+
9+
class TreeNodeIterator {
10+
std::stack<TreeNode *> stack_;
11+
12+
public:
13+
// Обязательными являются первые два параметра:
14+
15+
// Тип итератора:
16+
typedef std::input_iterator_tag iterator_category;
17+
// Тип выбирается из следующих типов:
18+
// input_iterator_tag
19+
// Должен поддерживать префиксную форму инкремента, оператор !=,
20+
// оператор * и оператор ->. Понадобится конструктор и конструктор
21+
// копирования.
22+
// output_iterator_tag
23+
// forward_iterator_tag
24+
// bidirectional_iterator_tag
25+
// random_access_iterator_tag
26+
27+
// Тип значения которое хранится и возвращается операторами * и ->:
28+
typedef TreeNode value_type;
29+
30+
// Тип который может описывать растояние между итераторами:
31+
typedef std::ptrdiff_t difference_type;
32+
33+
// Тип указателя на значение:
34+
typedef TreeNode *pointer;
35+
36+
// Тип ссылки на значения:
37+
typedef TreeNode &reference;
38+
39+
40+
explicit TreeNodeIterator(TreeNode *value = nullptr) {
41+
if (value)
42+
stack_.push(value);
43+
}
44+
45+
// TreeNode * operator*() const {
46+
// return !stack_.empty() ? stack_.top() : nullptr;
47+
// }
48+
49+
typename TreeNodeIterator::reference operator*() const {
50+
TreeNode *node = nullptr;
51+
if (!stack_.empty()) {
52+
node = stack_.top();
53+
}
54+
return *node;
55+
}
56+
57+
TreeNode * operator->() const {
58+
return !stack_.empty() ? stack_.top() : nullptr;
59+
}
60+
61+
62+
//  Хорошая практика добавлять поддержку != и == вместе:
63+
64+
bool operator==(TreeNodeIterator const &other) const {
65+
if (stack_.empty() || other.stack_.empty())
66+
return stack_.empty() && other.stack_.empty();
67+
68+
return stack_.top() == other.stack_.top();
69+
}
70+
71+
bool operator!=(TreeNodeIterator const &other) const {
72+
return !(*this == other);
73+
}
74+
75+
76+
TreeNodeIterator & operator++() {
77+
if (!stack_.empty()) {
78+
TreeNode *node = stack_.top();
79+
stack_.pop();
80+
for (int i = node->GetChildrenCount() - 1; i >= 0; --i)
81+
stack_.push(node->GetChildAtIndex(i));
82+
}
83+
return *this;
84+
}
85+
86+
};
87+
88+
#endif /* TreeNodeIterator_h */

it/tree-iter/main.cc

+57
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
#include <iostream>
2+
#include <cstdlib>
3+
4+
#include "TreeNode.h"
5+
#include "TreeNodeIterator.h"
6+
7+
8+
void test1() {
9+
std::vector<TreeNode> v = { };
10+
auto iter = v.begin();
11+
std::cout << (*iter).GetValue() << std::endl;
12+
++iter;
13+
while(iter != v.end()) {
14+
std::cout << iter->GetValue() << std::endl;
15+
++iter;
16+
}
17+
exit(0);
18+
}
19+
20+
int main(int argc, const char * argv[]) {
21+
// test1();
22+
23+
TreeNode root(0);
24+
25+
TreeNode n1(1);
26+
TreeNode n11(11);
27+
TreeNode n12(12);
28+
n1.AddChildNode(&n11);
29+
n1.AddChildNode(&n12);
30+
31+
TreeNode n2(2);
32+
TreeNode n21(21);
33+
TreeNode n22(22);
34+
n2.AddChildNode(&n21);
35+
n2.AddChildNode(&n22);
36+
37+
TreeNode n3(3);
38+
TreeNode n31(31);
39+
TreeNode n32(32);
40+
n3.AddChildNode(&n31);
41+
n3.AddChildNode(&n32);
42+
43+
root.AddChildNode(&n1);
44+
root.AddChildNode(&n2);
45+
root.AddChildNode(&n3);
46+
47+
auto iter = root.begin();
48+
std::cout << (*iter).GetValue() << std::endl;
49+
++iter;
50+
51+
while(iter != root.end()) {
52+
std::cout << iter->GetValue() << std::endl;
53+
++iter;
54+
}
55+
56+
return EXIT_SUCCESS;
57+
}

it/tree-iter/run.sh

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
#!/bin/bash
2+
3+
EXE_FILE="./program.out"
4+
5+
[ -f "$EXE_FILE" ] && rm "$EXE_FILE"
6+
7+
clang++ --std=c++17 `find . -maxdepth 1 -name "*.cc" | tr '\n' ' '` -o "$EXE_FILE" && "$EXE_FILE"

0 commit comments

Comments
 (0)