Nếu bạn muốn có thể gọi một hàm f
cho tất cả các loại có thành viên hàm getInt
, không chỉ X
, bạn có thể khai báo 2 quá tải cho hàm f
:
cho các loại có getInt
chức năng thành viên, bao gồm cả lớpX
cho tất cả các loại khác, bao gồm cả lớp Y
.
Giải pháp C ++ 11 / C ++ 17
Có suy nghĩ đó, bạn có thể làm một cái gì đó như thế này:
#include <iostream>
#include <type_traits>
template <typename, typename = void>
struct has_getInt : std::false_type {};
template <typename T>
struct has_getInt<T, std::void_t<decltype(((T*)nullptr)->getInt())>> : std::is_convertible<decltype(((T*)nullptr)->getInt()), int>
{};
class X {
public:
int getInt(){
return 9;
}
};
class Y {};
template <typename T,
typename std::enable_if<!has_getInt<T>::value, T>::type* = nullptr>
void f(T& v) {
// only for Y
std::cout << "Y" << std::endl;
}
template <typename T,
typename std::enable_if<has_getInt<T>::value, T>::type* = nullptr>
void f(T& v){
// only for X
int i = v.getInt();
std::cout << "X" << std::endl;
}
int main() {
X x;
f(x);
Y y;
f(y);
}
Kiểm tra nó trực tiếp .
Xin lưu ý rằng std::void_t
được giới thiệu trong C ++ 17, nhưng nếu bạn bị giới hạn ở C ++ 11, thì thực sự rất dễ thực hiện void_t
:
template <typename...>
using void_t = void;
Và đây là phiên bản C ++ 11 trực tiếp .
Chúng ta có gì trong C ++ 20?
C ++ 20 mang lại rất nhiều điều tốt và một trong số đó là các khái niệm . Trên đây là điều hợp lệ cho C ++ 11 / C ++ 14 / C ++ 17 có thể giảm đáng kể trong C ++ 20:
#include <iostream>
#include <concepts>
template<typename T>
concept HasGetInt = requires (T& v) { { v.getInt() } -> std::convertible_to<int>; };
class X {
public:
int getInt(){
return 9;
}
};
class Y {};
template <typename T>
void f(T& v) {
// only for Y
std::cout << "Y" << std::endl;
}
template <HasGetInt T>
void f(T& v){
// only for X
int i = v.getInt();
std::cout << "X" << std::endl;
}
int main() {
X x;
f(x);
Y y;
f(y);
}
Kiểm tra nó trực tiếp .
type_info
cấu trúc có một toán tử so sánh bằng , do đó,typeid(T) == typeid(X)
cũng nên làm việc.