一个高效的C++类型转换判断实现

最近在网上看到了一个C++类型转换判断的高效实现,分享出来共同学习。作者使用了sizeof关键词、函数重载与可变参数的功能,功能实现得简洁优雅。Talk is cheap, show me the code~

#include <iostream>
using namespace std;
template<class T, class U>
class CHECK_TYPE{
    class OTHER {char st[2];};
    static char Test(U);
    static OTHER Test(...);
    static T USAGE();
public:
    enum {result = sizeof(Test(USAGE())) == sizeof(char)};
};
class A {
};
class B : public A {
};
class C {
};
int main(){
    cout << CHECK_TYPE<double, int>::result << endl;
    cout << CHECK_TYPE<int, double>::result << endl;
    cout << CHECK_TYPE<B, A>::result << endl;
    cout << CHECK_TYPE<C, A>::result << endl;
}

分析:

  • 使用类模板生成某两个具体类型转换判断的具体类实现并在初始化的时候完成类型能否进行转化的结果;
  • 通过重载Test来表达出编译器对于类型的判断;
  • 借助于sizeof关键字的特性,只定义了函数声明表达返回类型,无需具体的定义,判断时也无需执行函数;
  • sizeof消耗最小,无函数调用,整个实现编译期已完成;而用typeinfo,typeid等需要解析,属于runtime。

代码中用到了可变参数。可变参数最典型的应用大概就是printf和scanf,原理是通过函数调用时候将类型、数目可变的参数全部压入函数调用栈,并定义了参数获取的宏来实现。以下面的函数定义为例,

void test(int count, ...)
{
    ... //do some stuff
}

由于函数参数压栈顺序是从右至左,所以在各个可变参数压入调用栈之后,count变量被压入相邻的低位栈地址(由于栈空间的使用由高位地址到低位地址增长)。使用传入的可变变量时,需要依赖count变量来初始化可变参数的初试地址。以上面函数的实现为例:

#include<stdarg.h>
//假设传入的可变参数都是整形变量,并且由count指示传入的可变参数的具体数量
void test(int count, ...)
{
    va_list st;
    va_start(st,count);
    for(int i = 0; i < count; ++i)
    {
        cout<<va_arg(st, int)<<endl;
    }
    va_end(ap);
}

其中使用va_list类型表示可变参数,并用到了三个宏,分别是va_start、va_arg、va_end来实现可变参数的访问。va_list在stdarg.h头文件中被定义为char *类型

typedef char * va_list;

而三个宏的定义如下(这里只找到了win平台的实现,Linux平台的实现应该与之类似):

#define va_start(ap,v) ( ap = (va_list)&v + _INTSIZEOF(v) )
#define va_end(ap) ( ap = (va_list)0 )
#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )

通过宏定义,不难看出

  • va_start负责初始化可变参数的访问地址;
  • va_end负责将指针置空;
  • va_arg负责根据类型长度访问ap一段内存地址取出类型为t的数据,并将指针后移;
    (1. ap += _INTSIZEOF(t):将ap指向下一个参数的地址)
    (2. *(t *)(ap - _INTSIZEOF(t)):得到当前参数地址,并强制类型转换成指定类型)
  • _INTSIZEOF负责地址的对齐问题

最后,回到文章题目,关于类型转换判断的功能,C11中有stl::is_convertable的新feature,但是这个版本的实现显得简洁易懂。

感谢code作者 yunchenglu

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,098评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,213评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,960评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,519评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,512评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,533评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,914评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,804评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,563评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,644评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,350评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,933评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,908评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,146评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,847评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,361评论 2 342

推荐阅读更多精彩内容