There was a problem: "write the second implementation of the template function", in which the same arguments are passed. That is, the signature of both functions completely coincides. The difference is that, as an argument, the std::vector containers and the std::map containers can be passed to the function. And depending on whether std::vector is or std::map, this or that implementation should be selected.
In this case, the SFINAE rule will be used, which states: If the final types of arguments cannot be calculated (to perform substitution of template parameters) of an overloaded template function, the compiler does not throw an error, but searches for another suitable overload. The error will be in three cases:
- No suitable overload was found.
- Several such overloads were found, and C ++ cannot decide which one to take.
- The overload was found, it turned out to be a template, and an error occurred when instantiating the template.
Our case will be the second. There are two overload template functions.
Implementation
To determine the required overload, std::enable_if will be used, which is used just for such cases to specialize a particular function depending on the type of data that needs to be processed.
The standard library STD, as it turned out, does not have the functionality to determine whether the container is std::map or not. But on StackOverflow this issue has already been discussed. The code given there needs only to be slightly corrected, and correctly applied to write a specialization of functions depending on the type of container.
The essence of the code is to decide at the compilation stage whether the element of the std::pair container is and, based on this, to decide whether the container is std::map. If the container is std::map, then select one function; in other cases, select the second implementation of the function.
#include <iterator> #include <type_traits> // templates for determining whether the std::pair container element is template <typename T> struct is_pair : std::false_type {}; template <typename T, typename U> struct is_pair<std::pair<T, U>> : std::true_type {}; template <typename T> constexpr bool is_pair_v = is_pair<T>::value; // templates for determining if the container is std::map template<typename, typename = void> struct is_mapping : std::false_type {}; template <typename Container> struct is_mapping<Container, std::enable_if_t<is_pair_v<typename std::iterator_traits<typename Container::iterator>::value_type>>> : std::true_type {}; template <typename T> constexpr bool is_mapping_v = is_mapping<T>::value; #include <map> #include <vector> #include <iostream> class ClassWithSpecializedMethods { public: ClassWithSpecializedMethods() {} // A specialized method for handling std::map template <class ContainerType> static typename std::enable_if<is_mapping_v<ContainerType>, void>::type printContainer(ContainerType& container); // A specialized method for handling std::vector template <class ContainerType> static typename std::enable_if<!is_mapping_v<ContainerType>, void>::type printContainer(ContainerType& container); }; // Method implementations template <class ContainerType> inline static typename std::enable_if<is_mapping_v<ContainerType>, void>::type ClassWithSpecializedMethods::printContainer(ContainerType& container) { std::cout << "Map:" << std::endl; for (const auto& [key, value] : container) { std::cout << "Key: " << key << " Value: " << value << std::endl; } std::cout << std::endl; } template <class ContainerType> inline static typename std::enable_if<!is_mapping_v<ContainerType>, void>::type ClassWithSpecializedMethods::printContainer(ContainerType& container) { std::cout << "Vector:" << std::endl; for (const auto& value : container) { std::cout << "Value: " << value << std::endl; } std::cout << std::endl; } // Testing functions int main() { std::cout << "is_pair:" << std::endl; std::cout << "Map: " << is_pair_v<std::iterator_traits<std::map<int, int>::iterator>::value_type> << std::endl; std::cout << "Vector: " << is_pair_v<std::iterator_traits<std::vector<int>::iterator>::value_type> << std::endl; std::cout << std::endl; std::cout << "is_mapping:" << std::endl; std::cout << "Map: " << is_mapping_v<std::map<int, int>> << std::endl; std::cout << "Vector: " << is_mapping_v<std::vector<int>> << std::endl; std::cout << std::endl; std::map<int, int> map_container = {{1, 1}, {2, 2}, {3, 3}}; std::vector<int> vector_container = {1, 2, 3}; ClassWithSpecializedMethods::printContainer(map_container); ClassWithSpecializedMethods::printContainer(vector_container); }
C++ standards used
For this code to work, you must use the C++17 standard; on a lesser standard, the code will not compile.
In QMake projects, the following lines should be added to the pro file
CONFIG += console c++17 QMAKE_CXXFLA ## Conclusion As a conclusion, we get the following conclusion ```lang-bsh is_pair: Map: 1 Vector: 0 is_mapping: Map: 1 Vector: 0 Map: Key: 1 Value: 1 Key: 2 Value: 2 Key: 3 Value: 3 Vector: Value: 1 Value: 2 Value: 3