The C++ Standard Template Library (STL) is a powerful and versatile collection of template classes and functions, designed to provide common data structures and algorithms. With the STL, C++ developers can efficiently manage collections of data and perform operations such as searching, sorting, and modifying data. By leveraging the STL, programmers can write more efficient, maintainable, and reusable code, significantly reducing development time.
Components of STL in C++
There are 4 components of STL in C++ are -
Algorithms
Containers
Functors
Iterators
Algorithms
Algorithms are a set of functions in the STL that perform operations on containers. These operations include searching, sorting, modifying, and manipulating data. Algorithms are designed to be generic, meaning they work with any container type, provided the container supports the required operations. Examples include std::sort, std::find, and std::for_each.
// Sorting the vector std::sort(vec.begin(), vec.end());
// Printing sorted vector for (int val : vec) { std::cout << val << " "; } return 0; }
Output:
1 2 3 4 5
Containers
Containers are data structures that store collections of objects. The STL provides various types of containers, each optimized for specific kinds of storage and access patterns. Common containers include std::vector, std::list, std::map, and std::set. Containers manage the memory allocation and deallocation for their elements, providing an easy-to-use and efficient way to handle collections.
// Accessing elements in the vector for (int val : vec) { std::cout << val << " "; } return 0; }
Output:
1 2 3 4 5
Functors
Functors (or function objects) are objects that can be called as if they were functions. They are created by overloading the operator(). Functors allow for more flexible and reusable code, especially when used with STL algorithms that accept function objects as parameters. They can maintain state between calls, unlike regular functions.
// Printing transformed vector for (int val : vec) { std::cout << val << " "; } return 0; }
Output:
3 6 9 12 15
Iterators
Iterators are objects that point to elements within a container. They provide a way to traverse and access elements in a container sequentially. Iterators are crucial for the operation of STL algorithms, allowing algorithms to work independently of the container types. Common iterator types include std::vector::iterator, std::list::iterator, and std::set::iterator.
// Using an iterator to access elements for (std::vector<int>::iterator it = vec.begin(); it != vec.end(); ++it) { std::cout << *it << " "; } return 0; }
Output:
10 20 30 40 50
C++ STL Containers
The STL contains sequence containers and associative containers. The containers are objects that store data. The standard sequence containers include vector, deque, and list. The standard associative containers are set, multiset, map, multimap, hash_set, hash_map, hash_multiset and hash_multimap. There are also container adaptors queue, priority_queue, and stack, that are containers with specific interface, using other containers as implementation.
Lets Discuss One By One With Proper Examples:
PAIR class template
std::pair template struct pair;
Pair of values
This class couples together with a pair of values, which may be of different types (T1 and T2). The individual values can be accessed through its public members first and second. Pairs are a particular case of the tuple.
Example:
include
using namespace std;
int main ()
{
pair pair1, pair3; //creats pair of integers
pair pair2; // creates pair of an integer an a string
pair1 = make_pair(1, 2); // insert 1 and 2 to the pair1
pair2 = make_pair(1, “Studytonight”) // insert 1 and “Studytonight” in pair2
pair3 = make_pair(2, 4)
cout<< pair1.first << endl; // prints 1, 1 being 1st element of pair1
cout<< pair2.second << endl; // prints Studytonight
if(pair1 == pair3)
cout<< “Pairs are equal” << endl;
else
cout<< “Pairs are not equal” << endl;
return 0;
}
* VECTOR
class template
<vector>
std::vector
template < class T, class Alloc = allocator<T> > class vector; // generic template
Vector
Vectors are sequence containers representing arrays that can change in size. Just like arrays, vectors use contiguous storage locations for their elements, which means that their elements can also be accessed using offsets on regular pointers to its elements, and just as efficiently as in arrays. But unlike arrays, their size can change dynamically, with their storage being handled automatically by the container.
Internally, vectors use a dynamically allocated array to store their elements. This array may need to be reallocated in order to grow in size when new elements are inserted, which implies allocating a new array and moving all elements to it.
Example:
include<iostream>
include<string>
include<vector>
int main() {
// Vector with 5 integers
// Default value of integers will be 0.
std::vector < int >
vecOfInts(5);
for (int x: vecOfInts)
std::cout << x << std::endl;
}
* LISTclass template
<list>
std::list
template < class T, class Alloc = allocator<T> > class list;
List: They are sequence containers that allow constant time insert and erase operations anywhere within the sequence, and iteration in both directions. List containers are implemented as doubly-linked lists; Doubly linked lists can store each of the elements they contain in different and unrelated storage locations. The ordering is kept internally by the association to each element of a link to the element preceding it and a link to the element following it.
Example: Creating a std::list of int and pushing elements in front and back std::list listOfNumbers; //Inserting elements at end in list listOfNumbers.push_back(5); listOfNumbers.push_back(6); //Inserting elements at front in list listOfNumbers.push_front(2); listOfNumbers.push_front(1);
DEQUEUE class template
std::deque template < class T, class Alloc = allocator > class deque;
Double-ended queuedeque (usually pronounced like “deck”) is an irregular acronym of double-ended queue. Double-ended queues are sequence containers with dynamic sizes that can be expanded or contracted on both ends (either it’s front or it’s back). Specific libraries may implement deques in different ways, generally as some form of a dynamic array. But in any case, they allow for the individual elements to be accessed directly through random access iterators, with storage handled automatically by expanding and contracting the container as needed.
Double ended queuedeque (usually pronounced like “deck”) is an irregular acronym of double-ended queue. Double-ended queues are sequence containers with dynamic sizes that can be expanded or contracted on both ends (either its front or its back).
Specific libraries may implement deques in different ways, generally as some form of dynamic array. But in any case, they allow for the individual elements to be accessed directly through random access iterators, with storage handled automatically by expanding and contracting the container as needed.
Example:
include<iostream>
include<deque>
using namespace std; void showdq(deque g) { deque :: iterator it; for (it = g.begin(); it != g.end(); ++it) cout << ‘\t’ << *it; cout << ‘\n’; } int main() { deque gquiz; gquiz.push_back(10); gquiz.push_front(20); gquiz.push_back(30); gquiz.push_front(15); cout << “The deque gquiz is : “; showdq(gquiz);
std::queue template > class queue; FIFO queue queues are a type of container adaptor, specifically designed to operate in a FIFO context (first-in first-out), where elements are inserted into one end of the container and extracted from the other.
Queues are implemented as containers adaptors, which are classes that use an encapsulated object of a specific container class as its underlying container, providing a specific set of member functions to access its elements. Elements are pushed into the “back” of the specific container and popped from its “front”. The underlying container may be one of the standard container class template or some other specifically designed container class. This underlying container shall support at least the following operations:
empty
size
front
back
push_back
pop_front
Example:
include<iostream>
include<queue>
using namespace std; int main() { queue queue1; queue1.emplace(1); queue1.emplace(2); queue1.emplace(3); if (queue1.empty()) { cout << “The queue is empty”; } else { cout << “The queue is not empty”; } return 0; }
PRIOITY QUEUE
class template
std::priority_queue
template ,
class Compare = less > class priority_queue;
Priority queue Priority queues are a type of container adaptors, specifically designed such that its first element is always the greatest of the elements it contains, according to some strict weak ordering criterion.
This context is similar to a heap, where elements can be inserted at any moment, and only the max heap element can be retrieved (the one at the top in the priority queue). Operations:- empty() • size() • front() • push_back() • pop_back()
std::stack template > class stack; LIFO stack Stacks are a type of container adaptor, specifically designed to operate in a LIFO context (last-in first-out), where elements are inserted and extracted only from one end of the container.
stacks are implemented as container adaptors, which are classes that use an encapsulated object of a specific container class as its underlying container, providing a specific set of member functions to access its elements. Elements are pushed/popped from the “back” of the specific container, which is known as the top of the stack.
The underlying container may be any of the standard container class templates or some other specifically designed container class. The container shall support the following operations: • empty • size • back • push_back • pop_back
Example: #include<iostream>
include<stack>
using namespace std; int main() { stack st; st.push(10); st.push(20); st.push(30); st.push(40);
st.pop();
st.pop();
while (!st.empty()) {
cout << ' ' << st.top();
st.pop();
}
}
SET class template
Set Sets are containers that store unique elements following a specific order.
In a set, the value of an element also identifies it (the value is itself the key, of type T), and each value must be unique. The value of the elements in a set cannot be modified once in the container (the elements are always const), but they can be inserted or removed from the container. Internally, the elements in a set are always sorted following a specific strict weak ordering criterion indicated by its internal comparison object (of type Compare).
Set containers are generally slower than unordered_set containers to access individual elements by their key, but they allow the direct iteration on subsets based on their order. Example: std::set template < class T, // set::key_type/value_type class Compare = less, // set::key_compare/value_compare class Alloc = allocator // set::allocator_type > class set;
MULTI SET class template
std::multiset Multiple-key set Multisets are containers that store elements following a specific order, and where multiple elements can have equivalent values.
In a multiset, the value of an element also identifies it (the value is itself the key, of type T). The value of the elements in a multiset cannot be modified once in the container (the elements are always const), but they can be inserted or removed from the container. Internally, the elements in a multiset are always sorted following a specific strict weak ordering criterion indicated by its internal comparison object (of type Compare).
Example: #include<iostream>
include<set>
include<iterator>
using namespace std; int main() { // empty multiset container multiset > gquiz1;
// insert elements in random order
gquiz1.insert(40);
gquiz1.insert(30);
gquiz1.insert(60);
gquiz1.insert(20);
gquiz1.insert(50);
gquiz1.insert(50); // 50 will be added again to the multiset unlike set
gquiz1.insert(10);
// printing multiset gquiz1
multiset <int, greater <int> > :: iterator itr;
cout << "\nThe multiset gquiz1 is : ";
for (itr = gquiz1.begin(); itr != gquiz1.end(); ++itr)
cout << "\nThe multiset gquiz2 after assign from gquiz1 is : ";
for (itr = gquiz2.begin(); itr != gquiz2.end(); ++itr)
{
cout << '\t' << *itr;
}
cout << endl;
// remove all elements up to element with value 30 in gquiz2
cout << "\ngquiz2 after removal of elements less than 30 : ";
gquiz2.erase(gquiz2.begin(), gquiz2.find(30));
for (itr = gquiz2.begin(); itr != gquiz2.end(); ++itr)
{
cout << '\t' << *itr;
}
// remove all elements with value 50 in gquiz2
int num;
num = gquiz2.erase(50);
cout << "\ngquiz2.erase(50) : ";
cout << num << " removed \t" ;
for (itr = gquiz2.begin(); itr != gquiz2.end(); ++itr)
{
cout << '\t' << *itr;
}
cout << endl;
//lower bound and upper bound for multiset gquiz1
cout << "gquiz1.lower_bound(40) : "
<< *gquiz1.lower_bound(40) << endl;
cout << "gquiz1.upper_bound(40) : "
<< *gquiz1.upper_bound(40) << endl;
//lower bound and upper bound for multiset gquiz2
cout << "gquiz2.lower_bound(40) : "
<< *gquiz2.lower_bound(40) << endl;
cout << "gquiz2.upper_bound(40) : "
<< *gquiz2.upper_bound(40) << endl;
return 0;
MAP class template
std::map Map Maps are associative containers that store elements formed by a combination of a key value and a mapped value, following a specific order.
In a map, the key values are generally used to sort and uniquely identify the elements, while the mapped values store the content associated with this key. The types of key and mapped value may differ and are grouped together in member type value_type, which is a pair type combining both:
typedef pair<const Key, T> value_type;
The mapped values in a map can be accessed directly by their corresponding key using the bracket operator ((operator[]).
Maps are typically implemented as binary search trees. Example: #include<iostream>
include<map>
using namespace std; int main () { map m{ {1,2} , {2,3} , {3,4} }; /* creates a map m with keys 1,2,3 and their corresponding values 2,3,4 / map map1; / creates a map with keys of type character and values of type integer */
map1["abc"]=100; // inserts key = "abc" with value = 100
map1["b"]=200; // inserts key = "b" with value = 200
map1["c"]=300; // inserts key = "c" with value = 300
map1["def"]=400; // inserts key = "def" with value = 400
map<char,int> map2 (map1.begin(), map1.end());
/* creates a map map2 which have entries copied
from map1.begin() to map1.end() */
map<char,int> map3 (m);
/* creates map map3 which is a copy of map m */
}
hash_set
hash_multiset
hash-map
hash_multimap
similar to a set, multiset, map, or multimap, respectively, but implemented using a hash_table; keys are not ordered, but a hash function must exist for the key type. These types were left out of the C++ standard; similar containers were standardized in C++, but with different names (unordered_set and unordered_map).
* BITSET
class template
<bitset>
std::bitset
template <size_t N> class bitset;
Bitset A bitset stores bits (elements with only two possible values: 0 or 1, true or false, …).The class emulates an array of bool elements, but optimised for space allocation: generally, each element occupies only one bit (which, on most systems, is eight times less than the smallest elemental type: char).
Each bit position can be accessed individually: for example, for a given bitset named foo, the expression foo[3] accesses its fourth bit, just like a regular array accesses its elements. But because no elemental type is a single bit in most C++ environments, the individual elements are accessed as special references type (see bitset:: reference).
Example:
include<bits/stdc++.h>
using namespace std; int main() { bitset<4> bset1(9); // bset1 contains 1001 bitset<4> bset2(3); // bset2 contains 0011
// comparison operator
cout << (bset1 == bset2) << endl; // false 0
cout << (bset1 != bset2) << endl; // true 1
// bitwise operation and assignment
cout << (bset1 ^= bset2) << endl; // 1010
cout << (bset1 &= bset2) << endl; // 0010
cout << (bset1 |= bset2) << endl; // 0011
// left and right shifting
cout << (bset1 <<= 2) << endl; // 1100
cout << (bset1 >>= 1) << endl; // 0110
// not operator
cout << (~bset2) << endl; // 1100
// bitwise operator
cout << (bset1 & bset2) << endl; // 0010
cout << (bset1 | bset2) << endl; // 0111
cout << (bset1 ^ bset2) << endl; // 0101
}
SORT function template
std::sort default (1) Sort elements in range Sorts the elements in the range [first,last) into ascending order.
The elements are compared using the operator< for the first version, and comp for the second. Example:
include<iostream>
include<algorithm>
using namespace std; void show(int a[]) { for(int i = 0; i < 10; ++i) cout << a[i] << ” “; } int main() { int a[10]= {1, 5, 8, 9, 6, 7, 3, 4, 2, 0}; cout << “\n The array before sorting is : “; show(a); sort(a, a+10); cout << “\n\n The array after sorting is : “; show(a); return 0; }
ValARRAY class template
std::valarray template class valarray; Valarray class A valarray object is designed to hold an array of values, and easily perform mathematical operations on them. It also allows special mechanisms to refer to subsets of elements in the arrays (see its operator[] overload).
Most mathematical operations can be applied directly to valarray objects, including arithmetical and comparison operators, affecting all its elements.
The valarray specification allows for libraries to implement it with several efficiency optimisations, such as parallelisation of certain operations, memory recycling or support for copy-on-reference / copy-on-write optimisations. Implementations may even replace valarray as the return type for standard functions described below, provided they behave as, and can be converted to, valarray objects.
Example:
// C++ code to demonstrate the working of
// apply() and sum()
include<iostream>
include<valarray> // for valarray functions
using namespace std;
int main()
{
// Initializing valarray
valarray varr = { 10, 2, 20, 1, 30 };
// Declaring new valarray
valarray<int> varr1 ;
// Using apply() to increment all elements by 5
varr1 = varr.apply([](int x){return x=x+5;});
// Displaying new elements value
cout << "The new valarray with manipulated values is : ";
for (int &x: varr1) cout << x << " ";
cout << endl;
// Displaying sum of both old and new valarray
cout << "The sum of old valarray is : ";
cout << varr.sum() << endl;
cout << "The sum of new valarray is : ";
cout << varr1.sum() << endl;
return 0;
}
Frequently Asked Questions
What is STL file in C++?
An STL file in C++ refers to the Standard Template Library, a collection of generic classes and functions for data structures and algorithms.
Is it good to use STL in C++?
Yes, using STL in C++ is beneficial as it provides efficient, tested, and reusable components for data handling and algorithm implementation.
Why is STL used?
STL is used to simplify and standardize data structure management and algorithm application, promoting code efficiency, maintainability, and reusability.
What is stack in C++ STL?
A stack in C++ STL is a container adapter that follows the LIFO (Last In, First Out) principle, allowing push and pop operations.
Conclusion
The C++ Standard Template Library (STL) is a cornerstone of modern C++ programming, offering a rich set of tools for efficient data management and algorithmic processing. By mastering the four core components—algorithms, containers, functors, and iterators—developers can significantly enhance their productivity and the quality of their code. The STL not only promotes code reuse and maintainability but also ensures high performance and reliability.