C++11模板:如何判断类中是否有指定名称的成员变量?

如何判断类中有指定的成员函数,网上可以找到不少的文章,比如下面这两篇就写得很详细了

《C++11之美》
《C++模板,判断是否存在成员函数,实现差异化操作 》

我现在关心的是如何判断一个类中有成员变量?成员变量有可能是数组,也可能是其他的类。
看了上面关于判断成员函数的文章,理解其原理后,判断是否成员变量,也是差不多的道理,实现起来非常简单:

/* 模板函数,检查T是否有名为's'的成员
 * value 为bool型检查结果
 * type为s成员的类型(value为true是有效)
 */
template<typename T>
struct has_member_s{
    template <typename _T>
    static auto check(_T)->typename std::decay<decltype(_T::s)>::type;
    static void check(...);
    using type=decltype(check(std::declval<T>()));
    enum{value=!std::is_void<type>::value};
};

上面这个模板是用来检查类中是否有名为s的成员,
以opencl中的cl_int2向量类型举例,下面是cl_int2的定义:

/* ---- cl_intn ---- */
typedef union
{
    cl_int  CL_ALIGNED(8) s[2];
#if __CL_HAS_ANON_STRUCT__
   __CL_ANON_STRUCT__ struct{ cl_int  x, y; };
   __CL_ANON_STRUCT__ struct{ cl_int  s0, s1; };
   __CL_ANON_STRUCT__ struct{ cl_int  lo, hi; };
#endif
#if defined( __CL_INT2__) 
    __cl_int2     v2;
#endif
}cl_int2;

可以看到,cl_int2中有个名为s数组。
下面是测试代码:

#include <iostream>
#include <CL/cl.hpp>
int main(int argc, char * argv[]){
	cout<<"cl_int2="<<has_member_s<cl_int2>::value<<endl;
	cout << "cl_int2=" << typeid(has_member_s<cl_int2>::type).name() << endl;
	cout<<"int="<<has_member_s<int>::value<<endl;//对比测试
}

gcc编译运行结果

cl_int2=1
cl_int2=Pi
int=0

vs2015编译运行结果

cl_int2=1
cl_int2=int * __ptr64
int=0

注意:
模板函数中这一句

static auto check(_T)->typename std::decay<decltype(_T::s)>::type;

decltype(_T::s)已经获取了_T::s的类型,用std::decay再套一层貌似是多余的,其实不然。
对于非数组成员变量,去掉std::decay这一层,直接写成

static auto check(_T)->decltype(_T::s);

是完全可以的(不论在gcc还是vs2015)。
但是对于数组类型的变量,上面的写法,在gcc下编译能通过,但运行结果错误。
大概gcc认为返回的值不能是int[2]这样的数组,只能是指针。

static auto check(_T)->cl_int[2]; // 不加`std::decay`时,返回数组,无效
static auto check(_T)->cl_int*; // 加上`std::decay`后,返回指针,有效

需要多次使用这个模板函数判断不同的成员变量时,用宏来改进上面的代码就显得很必要

/* 宏函数定义的模板函数,检查T是否有名为's'的成员
 * value 为bool型检查结果
 * type为s成员的类型(value为true是有效)
 */
#define has_member(s) \
template<typename T>\
struct has_member_##s{\
	template <typename _T>static auto check(_T)->typename std::decay<decltype(_T::s)>::type;\
	static void check(...);\
	using type=decltype(check(std::declval<T>()));\
	enum{value=!std::is_void<type>::value};\
};

将这个模板定义成宏后,如果要检查是否有s成员就以s为参数展开has_member

has_member(s);

如果要检查是否有x成员,就以x为参数展开has_member

has_member(x);
  • 6
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 6
    评论
### 回答1: 在C++中,模板类的静态成员变量的初始化必须在类的定义外进行。可以通过在类定义外部显式地初始化静态成员变量,或者使用静态成员函数来初始化静态成员变量。例如: ``` // 在类定义外部显式地初始化静态成员变量 template <typename T> class MyClass { public: static int static_member; }; template <typename T> int MyClass<T>::static_member = 0; // 使用静态成员函数来初始化静态成员变量 template <typename T> class MyClass { public: static int static_member; static int init_static_member() { static_member = 0; } }; template <typename T> int MyClass<T>::static_member = MyClass<T>::init_static_member(); ``` 需要注意的是,如果模板类的静态成员变量是一个类类型或者数组类型,那么需要使用类类型或者数组类型的默认构造函数进行初始化。 ### 回答2: 模板类的静态成员变量的初始化方式与普通类的静态成员变量相似,有以下几种方式: 1. 在类外部单独定义和初始化静态成员变量:可以在类的定义之外通过`类名<模板参数>::静态成员变量名 = 初始值`的方式对静态成员变量进行定义和初始化。 2. 在类内部通过静态成员函数进行初始化:可以在模板类中定义一个静态成员函数,在该函数内部对静态成员变量进行初始化。 3. 在模板类内部通过静态成员变量定义和初始化:可以在模板类内部直接定义静态成员变量,并利用模板参数进行初始化。 需要注意的是,在使用模板类时,不同的模板参数实例化出来的类都会拥有独立的静态成员变量。并且,静态成员变量的初始化只会在第一次使用时进行,而不是每次实例化模板类时都会进行初始化。 ### 回答3: 在C++中,模板类的静态成员变量的初始化稍显复杂。下面我将通过一些步骤来详细解释如何初始化模板类的静态成员变量。 首先,模板类的静态成员变量的定义应该放在类的外部,通常在头文件的同一位置,以便其他文件可以访问它。 其次,在类定义的前面加上"template<typename T>"来标识这是一个模板类。 然后,在模板类内部的定义之前加上关键字"template<typename T>",指定要初始化静态成员变量模板参数。 接下来,在类外部定义模板类的静态成员变量,使用与类定义中的模板参数相同的方式。 最后,在需要使用模板类的文件中,包括模板类的头文件,并根据具体类型进行实例化。 举个例子,假设有一个模板类TemplateClass,它有一个静态成员变量value。 首先,在TemplateClass的类定义前面加上"template<typename T>"。 然后,在类定义的内部之前加上"template<typename T>"。 接下来,在类外部定义模板类的静态成员变量:template<typename T> int TemplateClass<T>::value = 0; 最后,在需要使用TemplateClass的文件中,包括TemplateClass的头文件,并根据具体类型进行实例化:TemplateClass<int> obj;。 总结起来,初始化模板类的静态成员变量需要使用"template<typename T>"来标识模板类,并在类内外部根据具体类型进行定义和实例化。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

10km

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值