介绍一个C++奇巧淫技
你能实现这样一个函数吗:
MyType type;
HisType htype;
serialize_3(11, type, htype);
serialize_4(type, htype ,type, htype);
serialize_4(11, type , htype, htype);
参数类型自由,个数自由,怎么做呢?往下看:
[xiaochu.yh@OB macro]$ cat auto_type.cpp
/*
* (C) 1999-2013 Alibaba Inc.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*
*
* Version: auto_type.cpp, 09/04/2013 08:02:17 PM Yu Huang Exp $
*
* Author:
* Huang Yu <xiaochu.yh@alipay.com>
* Description:
* auto type match
*
*/ #include <stdio.h> void serialize()
{
return;
} class HisType
{
public:
HisType(const char *i) : value_(i) { }
~HisType() { }
void serialize() const
{
printf("HisType: s(%s)\n", value_);
}
private:
const char *value_;
}; class MyType
{
public:
MyType(int i) : value_(i) { }
~MyType() { }
void serialize() const
{
printf("MyType: f(%d)\n", value_);
}
private:
int value_;
}; void serialize(const int arg0)
{
printf("int: %d\n", arg0);
}
void serialize(const float arg0)
{
printf("float: %f\n", arg0);
} template<typename Arg0>
void serialize(const Arg0 &arg0)
{
arg0.serialize();
} template<typename Arg0>
void serialize_1(const Arg0 &arg0)
{
serialize(arg0);
} #define JOIN(x,y) JOIN2(x,y)
#define JOIN2(x,y) x##y #define DECVAL_1 0
#define DECVAL_2 1
#define DECVAL_3 2
#define DECVAL_4 3
#define DEC_VAL(n) DECVAL_##n // recursively expanding macro
#define ARG_TN0
#define ARG_TN1 typename Arg0
#define ARG_TN2 ARG_TN1, typename Arg1
#define ARG_TN3 ARG_TN2, typename Arg2
#define ARG_TN4 ARG_TN3, typename Arg3 #define ARG_PN0
#define ARG_PN1 const Arg0 & arg0
#define ARG_PN2 ARG_PN1, const Arg1 & arg1
#define ARG_PN3 ARG_PN2, const Arg2 & arg2
#define ARG_PN4 ARG_PN3, const Arg3 & arg3 #define ARG_AN0
#define ARG_AN1 arg0
#define ARG_AN2 ARG_AN1, arg1
#define ARG_AN3 ARG_AN2, arg2
#define ARG_AN4 ARG_AN3, arg3 #define ARG_CN0
#define ARG_CN1 arg0
#define ARG_CN2 arg1
#define ARG_CN3 arg2
#define ARG_CN4 arg3 #define SERIALIZE_DECLARE(NUM_ARG) \
template<JOIN(ARG_TN, NUM_ARG)> \
void JOIN(serialize_, NUM_ARG)(JOIN(ARG_PN, NUM_ARG)) SERIALIZE_DECLARE(2);
SERIALIZE_DECLARE(3);
SERIALIZE_DECLARE(4);
#define SERIALIZE_DEFINE(NUM_ARG) \
template<JOIN(ARG_TN, NUM_ARG)> \
void JOIN(serialize_, NUM_ARG)(JOIN(ARG_PN, NUM_ARG)) \
{ \
JOIN(serialize_, DEC_VAL(NUM_ARG))(JOIN(ARG_AN, DEC_VAL(NUM_ARG))); \
serialize(JOIN(ARG_CN, NUM_ARG)); \
} SERIALIZE_DEFINE(2);
SERIALIZE_DEFINE(3);
SERIALIZE_DEFINE(4); int main()
{
MyType type(4234);
HisType htype("home");
//先来个见面礼, 1是int类型,10.2f是float类型,type是自定义类型
serialize_4(1,10.2f,3, type);
printf("==============\n");
serialize_3(type,11, htype); // <== 注意下面的参数个数,以及参数顺序,完全自由!
printf("==============\n");
serialize_3(11 ,type, htype);
printf("==============\n");
serialize_3(htype ,type, htype);
printf("==============\n");
return 0;
}
编译运行结果:
[xiaochu.yh@OB macro]$ g++ auto_type.cpp
[xiaochu.yh@OB]$ ./a.out
int: 1
float: 10.200000
int: 3
MyType: f(4234)
==============
MyType: f(4234)
int: 11
HisType: s(home)
==============
int: 11
MyType: f(4234)
HisType: s(home)
==============
HisType: s(home)
MyType: f(4234)
HisType: s(home)
==============
该技术是从曲山同学的代码中学习来的,曲山对宏的运用真是炉火纯青!这里最神奇的就是下面一段代码,至今不明:
#define JOIN(x,y) JOIN2(x,y)
#define JOIN2(x,y) x##y
JOIN和JOIN2不是等价的吗?不过还真不是。如果只写JOIN2,在宏展开阶段会有比较诡异的事情发生。不信你试试。但是为什么呢?我也不知道。@曲山,求助啊~~
更全面的代码见OceanBase源码oceanbase/src/common/ob_rpc_stub.h和oceanbase/src/common/ob_rpc_macros.h
=============================================
UPDATE:
这篇帖子发到了内网,得到了@探晴同学指点,加上@元启 同学的解释,基本弄明白了JOIN的机制。
原因的确很简单。
#define MY_VALUE 2
#define JOIN(A,B) A##B
JOIN(hello, world)的输出结果就是 helloworld,
JOIN(MY_VALUE, b)的输出结果就是 MY_VALUEb。尽管MY_VALUE是个宏,你期待它展开成2b。
如何成为一个2b呢? 这么做:
#define JOIN(a, b) JOIN_EXPAND_PARAM(a,b)
#define JOIN_EXPAND_PARAM(a,b) a##b
JOIN(MY_VALUE, b)的展开过程是:
1. JOIN(MY_VALUE, b)展开成 JOIN_EXPAND_PARAM(2, b)
2. JOIN_EXPAND_PARAM(2, b) 展开成 2b
这句话:Macro arguments are completely macro-expanded before they are substituted into a macro body, unless they are stringified or pasted with other tokens.
参考: http://gcc.gnu.org/onlinedocs/cpp/Argument-Prescan.html
介绍一个C++奇巧淫技的更多相关文章
- LLDB奇巧淫技
打印视图层级 这个相信很多人都会了,是ta是ta就是ta recursiveDescription 用法大概就是如下 123 po [self.view recursiveDescription] p ...
- octave之奇巧淫技向量化计算实现寻找样本点所属聚类下标
前面有文章提到过,K-means算法,第一步骤是找出样本点的的所属聚类.下面用两种方式实现,一种是普通的循环,一种是完全向量化计算. 假设 : X 是m×n样本矩阵,其每一行是一个样本,m表示样本数目 ...
- iOS开发的一些奇巧淫技(转载)
iOS开发的一些奇巧淫技 http://www.cocoachina.com/ios/20141229/10783.html iOS开发的一些奇巧淫技2 http://www.cocoachina.c ...
- [异常解决] 奇巧淫技——VirtualBox中的linux无显示启动,并在win7上远程控制
楼主是资深技术宅(癖),由于感觉手上的老笔记本太卡,遂狠心买了个性能至强的主机同时配了个投影仪(满足躺着打代码的意淫场景).但是体验了大概一个月发现还是坐着打代码舒服,但是如下图坐着打代码总是要抬头看 ...
- iOS开发的一些奇巧淫技
TableView不显示没内容的Cell怎么办? 类似这种,我不想让下面那些空的显示. 很简单. self.tableView.tableFooterView = [[UIView alloc] in ...
- C基础 那些年用过的奇巧淫技
引言 - 为寻一颗明星 为要寻一颗明星 徐志摩 1924年12月1日<晨报六周年纪念增刊> 我骑著一匹拐腿的瞎马, 向著黑夜里加鞭:—— 向著黑夜里加鞭, 我跨著一匹拐腿的瞎马.// 我冲 ...
- iOS开发的一些奇巧淫技2
能不能只用一个pan手势来代替UISwipegesture的各个方向? - (void)pan:(UIPanGestureRecognizer *)sender { typedef NS_ENUM(N ...
- Windows的奇巧淫技(为什么GIF显示不出来??)
谁的电脑里没点小秘密?东藏西藏到最后自己都找不到了有木有?今天教大家个隐藏文件的高招: 将任意文件隐藏到图片中!怎么样?再也不用建什么「马列主义哲学」的文件夹啦!
- powerdesigner奇淫技
在日常开发中数据库的设计常常需要建立模型,而powerdesigner是个不错的选择.但很多时候用powerdesigner生成模型后再去创建表结构,会觉得烦和别扭.那么能不能数据库表建好后再生成模型 ...
随机推荐
- cocos2d-x on wp8架构简单介绍
1,基于C++的开发架构 支持3大移动平台以及3大桌面平台. 分为图形,声音,物理3大模块,另外还有脚本的导出. 在wp8/win32上的图形是基于d3d的,而在其它平台是基于opengl/openg ...
- 微信或手机浏览器在线显示office文件(已測试ios、android)
近期开发微信企业号,发现微信andriod版内置浏览器在打开文件方面有问题,可是ios版没有问题.原因是ios版使用的是safari浏览器 支持文档直接打开.可是andriod版使用的是腾讯浏览器x5 ...
- codeforces 148D 概率DP
题意: 原来袋子里有w仅仅白鼠和b仅仅黑鼠 龙和王妃轮流从袋子里抓老鼠. 谁先抓到白色老师谁就赢. 王妃每次抓一仅仅老鼠,龙每次抓完一仅仅老鼠之后会有一仅仅老鼠跑出来. 每次抓老鼠和跑出来的老鼠都是随 ...
- 学IT技术 轻松高薪就业
如今的社会是大鱼吃小鱼的时代,假设你没有过强的技术,是非常难在社会上立足,更不要谈占有一席之地了.假设你想学一门好技术,那你想知道如今学什么技术好吗?我想这要看如今市场须要什么人才,缺什么人才.同一时 ...
- Eclipse用法和技巧十三:自动生成的TODO注释1
使用eclipse的快捷键自动生成的代码,经常有这样的注释. 一眼看上去这个注释和一般的注释并无什么差别,不过TODO这个字符串的颜色不一样,应该有些内容.TODO是eclipse中提供的一种任务标签 ...
- 重操JS旧业第八弹:面向对象与继承
js里面没有语言语法层面的继承机制,但这并不意味着js就不能实现继承,利用js属性和方法动态性来模拟实现继承,通过总结大概有如下方法实现: 1 原型链继承 我们知道原型在对象中扮演着重要的角色,函数本 ...
- Control.Invoke和Control.BeginInvoke
问题的引入 下面有个简单的demo,大家一看代码就知道效果如何示例.我新建一个winform的程序,然后写入了如下代码: using System; using System.Windows.Form ...
- POJ 2594 Treasure Exploration(最小路径覆盖变形)
POJ 2594 Treasure Exploration 题目链接 题意:有向无环图,求最少多少条路径能够覆盖整个图,点能够反复走 思路:和普通的最小路径覆盖不同的是,点能够反复走,那么事实上仅仅要 ...
- ASP.NET - 记录错误日志
不需要像log4net/Nlog/Common Logging配置,简单好用. 不用增加声明logger对象,可记录当前执行状况. 可以定义 维护功能模板的开发人员,以便用功能模块对于开发人员. 出处 ...
- vs2012 不显示最近项目
visual studio起始页不显示最近使用项目的解决办法方法一 1.开始 → 运行 → 输入 regedit 回车,打开注册表编辑器. 2.定位到 HKEY_CURRENT_USER/Softwa ...