\chapter{Data structures} \index{data structure} A \key{data structure} is a way to store data in the memory of the computer. It is important to choose an appropriate data structure for a problem, because each data structure has its own advantages and disadvantages. The crucial question is: which operations are efficient in the chosen data structure? This chapter introduces the most important data structures in the C++ standard library. It is a good idea to use the standard library whenever possible, because it will save a lot of time. Later in the book we will learn more sophisticated data structures that are not available in the standard library. \section{Dynamic array} \index{dynamic array} \index{vector} A \key{dynamic array} is an array whose size can be changed during the execution of the program. The most popular dynamic array in C++ is the \texttt{vector} structure, that can be used almost like a regular array. The following code creates an empty vector and adds three elements to it: \begin{lstlisting} vector v; v.push_back(3); // [3] v.push_back(2); // [3,2] v.push_back(5); // [3,2,5] \end{lstlisting} After this, the elements can be accessed like in a regular array: \begin{lstlisting} cout << v[0] << "\n"; // 3 cout << v[1] << "\n"; // 2 cout << v[2] << "\n"; // 5 \end{lstlisting} The function \texttt{size} returns the number of elements in the vector. The following code iterates through the vector and prints all elements in it: \begin{lstlisting} for (int i = 0; i < v.size(); i++) { cout << v[i] << "\n"; } \end{lstlisting} \begin{samepage} A shorter way to iterate trough a vector is as follows: \begin{lstlisting} for (auto x : v) { cout << x << "\n"; } \end{lstlisting} \end{samepage} The function \texttt{back} returns the last element in the vector, and the function \texttt{pop\_back} removes the last element: \begin{lstlisting} vector v; v.push_back(5); v.push_back(2); cout << v.back() << "\n"; // 2 v.pop_back(); cout << v.back() << "\n"; // 5 \end{lstlisting} The following code creates a vector with five elements: \begin{lstlisting} vector v = {2,4,2,5,1}; \end{lstlisting} Another way to create a vector is to give the number of elements and the initial value for each element: \begin{lstlisting} // size 10, initial value 0 vector v(10); \end{lstlisting} \begin{lstlisting} // size 10, initial value 5 vector v(10, 5); \end{lstlisting} The internal implementation of the vector uses a regular array. If the size of the vector increases and the array becomes too small, a new array is allocated and all the elements are moved to the new array. However, this does not happen often and the average time complexity of \texttt{push\_back} is $O(1)$. \index{string} The \texttt{string} structure is also a dynamic array that can be used almost like a vector. In addition, there is special syntax for strings that is not available in other data structures. Strings can be combined using the \texttt{+} symbol. The function $\texttt{substr}(k,x)$ returns the substring that begins at index $k$ and has length $x$, and the function $\texttt{find}(\texttt{t})$ finds the position of the first occurrence of a substring \texttt{t}. The following code presents some string operations: \begin{lstlisting} string a = "hatti"; string b = a+a; cout << b << "\n"; // hattihatti b[5] = 'v'; cout << b << "\n"; // hattivatti string c = b.substr(3,4); cout << c << "\n"; // tiva \end{lstlisting} \section{Set structure} \index{set} A \key{set} is a data structure that maintains a collection of elements. The basic operations in a set are element insertion, search and removal. C++ contains two set implementations: \texttt{set} and \texttt{unordered\_set}. The structure \texttt{set} is based on a balanced binary tree and the time complexity of its operations is $O(\log n)$. The structure \texttt{unordered\_set} uses a hash table, and the time complexity of its operations is $O(1)$ on average. The choice which set implementation to use is often a matter of taste. The benefit in the \texttt{set} structure is that it maintains the order of the elements and provides functions that are not available in \texttt{unordered\_set}. On the other hand, \texttt{unordered\_set} is often more efficient. The following code creates a set that consists of integers, and shows some of the operations. The function \texttt{insert} adds an element to the set, the function \texttt{count} returns the number of occurrences of an element, and the function \texttt{erase} removes an element from the set. \begin{lstlisting} set s; s.insert(3); s.insert(2); s.insert(5); cout << s.count(3) << "\n"; // 1 cout << s.count(4) << "\n"; // 0 s.erase(3); s.insert(4); cout << s.count(3) << "\n"; // 0 cout << s.count(4) << "\n"; // 1 \end{lstlisting} A set can be used mostly like a vector, but it is not possible to access the elements using the \texttt{[]} notation. The following code creates a set, prints the number of elements in it, and then iterates through all the elements: \begin{lstlisting} set s = {2,5,6,8}; cout << s.size() << "\n"; // 4 for (auto x : s) { cout << x << "\n"; } \end{lstlisting} An important property of a set is that all the elements are \emph{distinct}. Thus, the function \texttt{count} always returns either 0 (the element is not in the set) or 1 (the element is in the set), and the function \texttt{insert} never adds an element to the set if it is already in the set. The following code illustrates this: \begin{lstlisting} set s; s.insert(5); s.insert(5); s.insert(5); cout << s.count(5) << "\n"; // 1 \end{lstlisting} C++ also has the structures \texttt{multiset} and \texttt{unordered\_multiset} that work otherwise like \texttt{set} and \texttt{unordered\_set} but they can contain multiple instances of an element. For example, in the following code all three instances of the number 5 are added to the set: \begin{lstlisting} multiset s; s.insert(5); s.insert(5); s.insert(5); cout << s.count(5) << "\n"; // 3 \end{lstlisting} The function \texttt{erase} removes all instances of an element from a \texttt{multiset}: \begin{lstlisting} s.erase(5); cout << s.count(5) << "\n"; // 0 \end{lstlisting} Often, only one instance should be removed, which can be done as follows: \begin{lstlisting} s.erase(s.find(5)); cout << s.count(5) << "\n"; // 2 \end{lstlisting} \section{Map structure} \index{map} A \key{map} is a generalized array that consists of key-value-pairs. While the keys in a regular array are always the consecutive integers $0,1,\ldots,n-1$, where $n$ is the size of the array, the keys in a map can be of any data type and they do not have to be consecutive values. C++ contains two map implementations that correspond to the set implementations: the structure \texttt{map} is based on a balanced binary tree and accessing an element takes $O(\log n)$ time, while the structure \texttt{unordered\_map} uses a hash map and accessing an element takes $O(1)$ time on average. The following code creates a map where the keys are strings and the values are integers: \begin{lstlisting} map m; m["monkey"] = 4; m["banana"] = 3; m["harpsichord"] = 9; cout << m["banana"] << "\n"; // 3 \end{lstlisting} If the value of a key is requested but the map does not contain it, the key is automatically added to the map with a default value. For example, in the following code, the key ''aybabtu'' with value 0 is added to the map. \begin{lstlisting} map m; cout << m["aybabtu"] << "\n"; // 0 \end{lstlisting} The function \texttt{count} determines if a key exists in the map: \begin{lstlisting} if (m.count("aybabtu")) { cout << "key exists in the map"; } \end{lstlisting} The following code prints all keys and values in the map: \begin{lstlisting} for (auto x : m) { cout << x.first << " " << x.second << "\n"; } \end{lstlisting} \section{Iterators and ranges} \index{iterator} Many functions in the C++ standard library operate with iterators. An \key{iterator} is a variable that points to an element in a data structure. Often used iterators are \texttt{begin} and \texttt{end} that define a range that contains all elements in a data structure. The iterator \texttt{begin} points to the first element in the data structure, and the iterator \texttt{end} points to the position \emph{after} the last element. The situation looks as follows: \begin{center} \begin{tabular}{llllllllll} \{ & 3, & 4, & 6, & 8, & 12, & 13, & 14, & 17 & \} \\ & $\uparrow$ & & & & & & & & $\uparrow$ \\ & \multicolumn{3}{l}{\texttt{s.begin()}} & & & & & & \texttt{s.end()} \\ \end{tabular} \end{center} Note the asymmetry in the iterators: \texttt{s.begin()} points to an element in the data structure, while \texttt{s.end()} points outside the data structure. Thus, the range defined by the iterators is \emph{half-open}. \subsubsection{Working with ranges} Iterators are used in C++ standard library functions that work with ranges of data structures. Usually, we want to process all elements in a data structure, so the iterators \texttt{begin} and \texttt{end} are given for the function. For example, the following code sorts a vector using the function \texttt{sort}, then reverses the order of the elements using the function \texttt{reverse}, and finally shuffles the order of the elements using the function \texttt{random\_shuffle}. \index{sort@\texttt{sort}} \index{reverse@\texttt{reverse}} \index{random\_shuffle@\texttt{random\_shuffle}} \begin{lstlisting} sort(v.begin(), v.end()); reverse(v.begin(), v.end()); random_shuffle(v.begin(), v.end()); \end{lstlisting} These functions can also be used with a regular array. In this case, the functions are given pointers to the array instead of iterators: \newpage \begin{lstlisting} sort(t, t+n); reverse(t, t+n); random_shuffle(t, t+n); \end{lstlisting} \subsubsection{Set iterators} Iterators are often used when accessing elements in a set. The following code creates an iterator \texttt{it} that points to the first element in the set: \begin{lstlisting} set::iterator it = s.begin(); \end{lstlisting} A shorter way to write the code is as follows: \begin{lstlisting} auto it = s.begin(); \end{lstlisting} The element to which an iterator points can be accessed through the \texttt{*} symbol. For example, the following code prints the first element in the set: \begin{lstlisting} auto it = s.begin(); cout << *it << "\n"; \end{lstlisting} Iterators can be moved using the operators \texttt{++} (forward) and \texttt{---} (backward), meaning that the iterator moves to the next or previous element in the set. The following code prints all elements in the set: \begin{lstlisting} for (auto it = s.begin(); it != s.end(); it++) { cout << *it << "\n"; } \end{lstlisting} The following code prints the last element in the set: \begin{lstlisting} auto it = s.end(); it--; cout << *it << "\n"; \end{lstlisting} The function $\texttt{find}(x)$ returns an iterator that points to an element whose value is $x$. However, if the set does not contain $x$, the iterator will be \texttt{end}. \begin{lstlisting} auto it = s.find(x); if (it == s.end()) cout << "x is missing"; \end{lstlisting} The function $\texttt{lower\_bound}(x)$ returns an iterator to the smallest element in the set whose value is \emph{at least} $x$, and the function $\texttt{upper\_bound}(x)$ returns an iterator to the smallest element in the set whose value is \emph{larger than} $x$. If such elements do not exist, the return value of the functions will be \texttt{end}. These functions are not supported by the \texttt{unordered\_set} structure that does not maintain the order of the elements. \begin{samepage} For example, the following code finds the element nearest to $x$: \begin{lstlisting} auto a = s.lower_bound(x); if (a == s.begin() && a == s.end()) { cout << "the set is empty\n"; } else if (a == s.begin()) { cout << *a << "\n"; } else if (a == s.end()) { a--; cout << *a << "\n"; } else { auto b = a; b--; if (x-*b < *a-x) cout << *b << "\n"; else cout << *a << "\n"; } \end{lstlisting} The code goes through all possible cases using the iterator \texttt{a}. First, the iterator points to the smallest element whose value is at least $x$. If \texttt{a} is both \texttt{begin} and \texttt{end} at the same time, the set is empty. If \texttt{a} equals \texttt{begin}, the corresponding element is nearest to $x$. If \texttt{a} equals \texttt{end}, the last element in the set is nearest to $x$. If none of the previous cases holds, the element nearest to $x$ is either the element that corresponds to $a$ or the previous element. \end{samepage} \section{Other structures} \subsubsection{Bitset} \index{bitset} A \texttt{bitset} is an array where each element is either 0 or 1. For example, the following code creates a bitset that contains 10 elements: \begin{lstlisting} bitset<10> s; s[2] = 1; s[5] = 1; s[6] = 1; s[8] = 1; cout << s[4] << "\n"; // 0 cout << s[5] << "\n"; // 1 \end{lstlisting} The benefit in using a bitset is that it requires less memory than a regular array, because each element in the bitset only uses one bit of memory. For example, if $n$ bits are stored as an \texttt{int} array, $32n$ bits of memory will be used, but a corresponding bitset only requires $n$ bits of memory. In addition, the values in a bitset can be efficiently manipulated using bit operators, which makes it possible to optimize algorithms. The following code shows another way to create a bitset: \begin{lstlisting} bitset<10> s(string("0010011010")); cout << s[4] << "\n"; // 0 cout << s[5] << "\n"; // 1 \end{lstlisting} The function \texttt{count} returns the number of ones in the bitset: \begin{lstlisting} bitset<10> s(string("0010011010")); cout << s.count() << "\n"; // 4 \end{lstlisting} The following code shows examples of using bit operations: \begin{lstlisting} bitset<10> a(string("0010110110")); bitset<10> b(string("1011011000")); cout << (a&b) << "\n"; // 0010010000 cout << (a|b) << "\n"; // 1011111110 cout << (a^b) << "\n"; // 1001101110 \end{lstlisting} \subsubsection{Deque} \index{deque} A \texttt{deque} is a dynamic array whose size can be changed at both ends of the array. Like a vector, a deque contains functions \texttt{push\_back} and \texttt{pop\_back}, but it also contains additional functions \texttt{push\_front} and \texttt{pop\_front} that are not available in a vector. A deque can be used as follows: \begin{lstlisting} deque d; d.push_back(5); // [5] d.push_back(2); // [5,2] d.push_front(3); // [3,5,2] d.pop_back(); // [3,5] d.pop_front(); // [5] \end{lstlisting} The internal implementation of a deque is more complex than the implementation of a vector. For this reason, a deque is slower than a vector. Still, the time complexity of adding and removing elements is $O(1)$ on average at both ends. \subsubsection{Stack} \index{stack} A \texttt{stack} is a data structure that provides two $O(1)$ time operations: adding an element to the top, and removing an element from the top. It is only possible to access the top element of a stack. The following code shows how a stack can be used: \begin{lstlisting} stack s; s.push(3); s.push(2); s.push(5); cout << s.top(); // 5 s.pop(); cout << s.top(); // 2 \end{lstlisting} \subsubsection{Queue} \index{queue} A \texttt{queue} also provides two $O(1)$ time operations: adding a element to the end of the queue, and removing the first element in the queue. It is only possible to access the first and the last element of a queue. The following code shows how a queue can be used: \begin{lstlisting} queue s; s.push(3); s.push(2); s.push(5); cout << s.front(); // 3 s.pop(); cout << s.front(); // 2 \end{lstlisting} \subsubsection{Priority queue} \index{priority queue} \index{heap} A \texttt{priority\_queue} maintains a set of elements. The supported operations are insertion and, depending on the type of the queue, retrieval and removal of either the minimum element or the maximum element. The time complexity is $O(\log n)$ for insertion and removal and $O(1)$ for retrieval. While a set structure efficiently supports all the operations of a priority queue, the benefit in using a priority queue is that it has smaller constant factors. A priority queue is usually implemented using a heap structure that is much simpler than a balanced binary tree needed for an ordered set. \begin{samepage} As default, the elements in the C++ priority queue are sorted in decreasing order, and it is possible to find and remove the largest element in the queue. The following code shows an example: \begin{lstlisting} priority_queue q; q.push(3); q.push(5); q.push(7); q.push(2); cout << q.top() << "\n"; // 7 q.pop(); cout << q.top() << "\n"; // 5 q.pop(); q.push(6); cout << q.top() << "\n"; // 6 q.pop(); \end{lstlisting} \end{samepage} Using the following declaration, we can create a priority queue that supports finding and removing the minimum element: \begin{lstlisting} priority_queue,greater> q; \end{lstlisting} \section{Comparison to sorting} Often it is possible to solve a problem using either data structures or sorting. Sometimes there are remarkable differences in the actual efficiency of these approaches, which may be hidden in their time complexities. Let us consider a problem where we are given two lists $A$ and $B$ that both contain $n$ integers. Our task is to calculate the number of integers that belong to both of the lists. For example, for the lists \[A = [5,2,8,9,4] \hspace{10px} \textrm{and} \hspace{10px} B = [3,2,9,5],\] the answer is 3 because the numbers 2, 5 and 9 belong to both of the lists. A straightforward solution to the problem is to go through all pairs of numbers in $O(n^2)$ time, but next we will concentrate on more efficient solutions. \subsubsection{Solution 1} We construct a set of the numbers in $A$, and after this, we iterate through the numbers in $B$ and check for each number if it also belongs to $A$. This is efficient because the numbers in $A$ are in a set. Using the \texttt{set} structure, the time complexity of the algorithm is $O(n \log n)$. \subsubsection{Solution 2} It is not needed to maintain an ordered set, so instead of the \texttt{set} structure we can also use the \texttt{unordered\_set} structure. This is an easy way to make the algorithm more efficient, because we only have to change the underlying data structure. The time complexity of the new algorithm is $O(n)$. \subsubsection{Solution 3} Instead of data structures, we can use sorting. First, we sort both lists $A$ and $B$. After this, we iterate through both the lists at the same time and find the common elements. The time complexity of sorting is $O(n \log n)$, and the rest of the algorithm works in $O(n)$ time, so the total time complexity is $O(n \log n)$. \subsubsection{Efficiency comparison} The following table shows how efficient the above algorithms are when $n$ varies and the elements in the lists are random integers between $1 \ldots 10^9$: \begin{center} \begin{tabular}{rrrr} $n$ & solution 1 & solution 2 & solution 3 \\ \hline $10^6$ & $1{,}5$ s & $0{,}3$ s & $0{,}2$ s \\ $2 \cdot 10^6$ & $3{,}7$ s & $0{,}8$ s & $0{,}3$ s \\ $3 \cdot 10^6$ & $5{,}7$ s & $1{,}3$ s & $0{,}5$ s \\ $4 \cdot 10^6$ & $7{,}7$ s & $1{,}7$ s & $0{,}7$ s \\ $5 \cdot 10^6$ & $10{,}0$ s & $2{,}3$ s & $0{,}9$ s \\ \end{tabular} \end{center} Solutions 1 and 2 are equal except that they use different set structures. In this problem, this choice has an important effect on the running time, because solution 2 is 4–5 times faster than solution 1. However, the most efficient solution is solution 3 that uses sorting. It only uses half of the time compared to solution 2. Interestingly, the time complexity of both solution 1 and solution 3 is $O(n \log n)$, but despite this, solution 3 is ten times faster. This can be explained by the fact that sorting is a simple procedure and it is done only once at the beginning of solution 3, and the rest of the algorithm works in linear time. On the other hand, solution 3 maintains a complex balanced binary tree during the whole algorithm.