在懂得BREW接口的原理之后, 那么该知道BREW接口是如何声明和实现了
参考:http://blog.csdn.net/peteryxk/article/details/1584514
首先介绍几个用到的宏定义:
l #define VTBL(iname) iname##Vtbl
例:VTBL(IWindow)将被替换为 IWindowVtbl。
从名字的后缀可以看出,它是模拟C++的虚函数的函数表。表中的每一项代表了一个函数指针。通过给指针赋予不同的值,便可以得到同一接口的不同实现。
l #define AEEVTBL(iname) iname##Vtbl
该宏的作用和第一个宏完全一样。唯一不同的是第一个宏用于第一种风格的接口声明,而AEEVTBL用于第二种风格的接口的声明。见下。
l #define INHERIT_IBase(iname) /
uint32 (*AddRef) (iname*);/
uint32 (*Release) (iname*)
例:INHERIT_IBASE(IWindow)将被替换为:
Uint32 (*AddRef) ( IWindow*) ;
Uint32 (*Release) (IWindow*)
从名字中可以看出,它是用来管理组件的生命周期的函数,回忆回忆COM的生命周期管理函数。
一,第一种风格的接口
1, 接口的声明
#define QINTERFACE(iname) struct _##iname {/
struct VTBL(iname) *pvt;/
};/
typedef struct VTBL(iname) VTBL(iname);/
struct VTBL(iname)
例:QINTERFACE(IWindow)将被替换为:
Struct _IWindow {
Struct IWindowVtbl *pvt ;
} ;
typedef struct IWindowVtbl IWindowVtbl ;
struct IWindowVtbl
如果在该声明的后面追加函数指针,如:
QINTERFACE(IWindow) {
INHERIT_IBASE(IWindow) ;
};
那么它将被替换为:
Struct _IWindow {
Struct IWindowVtbl *pvt ;
} ;
typedef struct IWindowVtbl IWindowVtbl ;
struct IWindowVtbl {
Uint32 (*AddRef) ( IWindow*) ;
Uint32 (*Release) (IWindow*)
} ;
由于此时IWindow并没有被声明为类型,所以在此之前需要它的类型定义。在下面的接口使用中,如果要组合该接口,那么需要IWindow的一个实例,所以IWindow的类型定义如下:
typedef struct _IWindow IWindow;
2, 接口的使用
如下为该接口的实现:
Struct WindowImpl {
IWindow vtIWindow;
Int nRef ;
…
} ;
_IWindow是一个结构体,该结构体的唯一成员为指向虚函数表IWindowVtbl的指针pvt。
有一个专门的宏用来访问该虚函数表指针。
#define GET_PVTBL(p,iname) ((iname*)p)->pvt
WindowImpl都是通过malloc等在heap上分配的。
pObj = malloc ( sizeof( WindowImpl ) + sizeof( IWindowVtbl ) ) ;
pObj->pvt = pObj + 1 ;
即分配空间时,分配的大小为实现类的结构体大小加虚函数表体的大小,并设置虚函数表指针指向虚函数表。
当调用AddRef函数时,可以使用如下方法:
GET_PVTBL(pObj, IWindow)->AddRef();
它被替换为:
((IWindow*)pObj)->pvt->AddRef();
能够正确的调用AddRef函数。
为了方便起见,一般定义如下的宏:
#define IWINDOW_AddRef(p) Get_PVTBL(p, IWindow)->AddRef()
二,第二种风格的接口
1, 接口的声明
#define AEEINTERFACE(iname) /
typedef struct AEEVTBL(iname) AEEVTBL(iname); /
struct AEEVTBL(iname)
如果声明了:
AEEINTERFACE(IWindow) {
INHERIT_IBASE(IWindow) ;
};
那么将被替换为:
typedef struct IWindowVtbl IWindowVtbl ;
struct IWindowVtbl {
Uint32 (*AddRef) ( IWindow*) ;
Uint32 (*Release) (IWindow*) ;
} ;
同样在此之前,应首先声明IWindow。
更被推荐的做法如下:
l 首先进行宏定义INHERIT_XXX
例如:
#define INHERIT_IWindow(IWindow) INHERIT_IBASE(IWindow)
l 然后使用AEEINTERFACE_DEFINE定义类型
它的宏定义如下:
#define AEEINTERFACE_DEFINE(iname)/
typedef struct iname iname;/
AEEINTERFACE(iname) {/
INHERIT_##iname(iname);/
}
这样AEEINTERFACE_DEFIN(IWindow)将被替换为:
typedef struct IWindow IWindow ;
typedef struct IWindowVtbl IWindowVtbl ;
struct IWindowVtbl {
Uint32 (*AddRef) ( IWindow*) ;
Uint32 (*Release) (IWindow*) ;
} ;
注:如果仅仅有如下的类型定义:
typedef truct Dummy Dummy ;
那么程序中有如下的语句的话:Dummy * pDummy ; 是合法的。
但是 Dummy dummy将是不合法的。因为Dummy是不完整的类型,所以不能实例化。但是可以存在指向不完全类型的指针。
上述接口就是利用了这一点,它会在函数的内部将指针牵制类型转换为实现类的指针。
2, 接口的使用
如下为该接口的实现:
Struct WindowImpl {
AEEVTBL(IWindow) *pvt;
Int nRef ;
…
} ;
pvt是指向IWindowVtbl的指针,即虚函数表指针。
相应的宏为:#define AEEGETPVTBL(p,iname) (*((AEEVTBL(iname) **)((void *)p)))
假如pObj为指向WindowImpl的指针。那么AEEGETPVTBL(pObj,IWindow)将被替换为
(*(IWindowVtbl**)(void*)pObj),此表达式的结果就是指向IWindowVtbl结构体的指针。
AEEGETPVTBL(pObj,IWindow)->AddRef();便完成了函数的调用。
一般写一个辅助宏:
#define IWINDOW_AddRef(p) AEEGETPVTBL(p,IWindow)->AddRef()
三,两种风格间的比较
第一种风格,为了在实现类中保存vtbl的指针,必须借助于具体的类型IWindow。而在第二种风格中,IWindow仅仅是个不完整的类型定义。具体的在AeeInterface.h中有如下文字:
/*
||
|| There's a problem with QINTERFACE(): it wastes namespace and forces
|| implementers to choose a name for a type that's unnecessarily
|| different from the interface name, since the QINTERFACE macro
|| makes an interface into a concrete type with only one member(!),
|| the vtable.
||
|| The macros below have all the same type-safety of the QINTERFACE()
|| macros, but don't force a concrete type on the object, allowing
|| implementers of interfaces to use the same type names, thus
|| obviating the need to do a type cast on entry to a method.
||
|| It should be easy to redefine QINTERFACE and its descendants
|| in terms of the below.
||
*/
在懂得BREW接口的原理之后, 那么该知道BREW接口是如何声明和实现了的更多相关文章
- 函数式 js 接口实现原理,以及 lodash/fp 模块
函数式 js 接口 之前在 youtube 上看到一个技术视频,讲“underscore.js的接口为什么不好用”,以及什么样的接口更好用.演讲者是 lodash.js 的作者,他提出了一种“全面函数 ...
- IM开发基础知识补课:正确理解前置HTTP SSO单点登陆接口的原理
1.前言 一个安全的信息系统,合法身份检查是必须环节.尤其IM这种以“人”为中心的社交体系,身份认证更是必不可少. 一些PC时代小型IM系统中,身份认证可能直接做到长连接中(也就是整个IM系统都是以长 ...
- Mybatis接口编程原理分析(三)
前面两篇博客Mybatis接口编程原理分析(一)和Mybatis接口编程原理分析(二)我们介绍了MapperProxyFactory.MapperProxy和MapperMethod的操作及源码分析, ...
- Mybatis接口编程原理分析(二)
在上一篇博客中 Mybatis接口编程原理分析(一)中我们介绍了MapperProxyFactory和MapperProxy,接下来我们要介绍的是MapperMethod MapperMethod:它 ...
- 浅入浅出 Go 语言接口的原理
浅入浅出 Go 语言接口的原理 接口是 Go 语言的重要组成部分,它在 Go 语言中通过一组方法指定了一个对象的行为,接口 interface 的引入能够让我们在 Go 语言更好地组织并写出易于测试的 ...
- mybatis——mybatis打印sql 接口工作原理
https://blog.csdn.net/Lxinccode/article/details/79218566 接口工作原理: Dao接口即Mapper接口.接口的全限名,就是映射文件中的names ...
- Mybatis接口编程原理分析(一)
Mybatis接口编程示例 (1)首先定义接口编程需要的接口及其方法 public interface IUserMapper { public User getById(int id);//接口方法 ...
- 35.按要求编写Java程序: (1)编写一个接口:InterfaceA,只含有一个方法int method(int n); (2)编写一个类:ClassA来实现接口InterfaceA,实现int method(int n)接口方 法时,要求计算1到n的和; (3)编写另一个类:ClassB来实现接口InterfaceA,实现int method(int n)接口 方法时,要求计算n的阶乘(n
35.按要求编写Java程序: (1)编写一个接口:InterfaceA,只含有一个方法int method(int n): (2)编写一个类:ClassA来实现接口InterfaceA,实现in ...
- Ajax请求接口加密研究(针对网页前端的接口安全加密机制研究)
通常我们在h5前端调用后台接口时,一般是ajax,那么接口的安全成了一个问题. 这里可以肯定的说,前端调用的接口一定要验证! 然后剖析了微信网页版.京东网页版这些,也都是通过接口的形势绑定数据,所以在 ...
随机推荐
- version `GLIBC_2.14' not found 解决方法.
from http://blog.csdn.net/force_eagle/article/details/8684669 version `GLIBC_2.14' not found 解决方法. 一 ...
- GNU风格 ARM汇编语法2
.GNU汇编程序中的标号symbol(或label) 标号只能由a-z,A-Z,-,".",_等(由点.字母.数字.下划线等组成,除局部标号外,不能以数字开头)字符组成. Symb ...
- Office 2013 Excel 打开文档很慢很慢的解决方法
这个问题查了很多案例,试了很多方法,但是只有下面这个方法有用! 这几天打开excel文档很慢很慢,双击之后好久没反应,过会儿它才慢慢冒出来,当时将就了,刚刚休息的时候想着查一下吧,不然很影响工作效率! ...
- Python的sys.argv使用说明 通过终端写入环境变量
刚开始使用这个参数的时候,很不明白其含义.网上搜索很多都是贴的官网上面的一则实例,说看懂,就明白.可是,我看不懂.现在在回头看这个参数使用,并不是很麻烦. 举几个小例子就明白了. 创建一个脚本,内容如 ...
- 关于SVN提交时报out-of-date错误的解决方法
提交项目文件时,报如下的信息:Item is out-of-datesvn: Commit failed (details follow):svn: Item '/xxx/xxx/xxx/xxx/xx ...
- Linux 下 ps 命令
简述 Linux中的ps命令是Process Status的缩写.ps命令用来列出系统中当前运行的那些进程.ps命令列出的是当前那些进程的快照,就是执行ps命令的那个时刻的那些进程,动态的显示进程信息 ...
- hadoop中的Jobhistory历史服务器
1. 启动脚本 mr-jobhistory-daemon.sh start historyserver 2. 配置说明 jobhistory用于查询每个job运行完以后的历史日志信息,是作为一台单独 ...
- centsos各个版本的区别
CentOS-7.0-1406-x86_64-DVD.iso 标准安装版,一般下载这个就可以了CentOS-7.0-1406-x86_64-NetInstall.iso ...
- Javascript实现浏览器菜单命令
每当我们看到别人网页上的打开.打印.前进.另存为.后退.关闭本窗口.禁用右键等实现浏览器命令的链接,而自己苦于不能实现时,是不是感到很遗憾?是不是也想实现?如果能在网页上能实现浏览器的命令,将是多么有 ...
- Qt 反射
简介 本文主要讲解Qt是如何实现反射,以及一点点反射使用的小心得. 文章概览 Qt反射内幕小窥 详细内容 反射前期准备 得到注册的类成员变量 得到注册的类成员函数 访问类成员属性(get,set) 调 ...