为了多多研习ATL想到生成的代码,visio、rose、word、excel加上我的笔本全上阵,终于把各个模板类混了个了解、跟宏混了个脸熟。当我借助调试器想把一个classfactory和com对象的由来看个明白的时候,我发现疑惑很多。不过好在简单看出了点门道,但要说清楚,非三言两语可以完成,只好写个系列以AC1、AC2、ACn等标识。
__declspec(allocate("segname")) 告诉编译器分配一个名字为segname的段,譬如:
- __declspec(allocate("LIUAN")) int myValue = 100;
告诉编译器,分配一个叫LIUAN的段,把我们的变量myValue存放在那里,另一种说法是向符号表里添加myValue符号,使用了这个手段的项目在编译出来的map文件里,可以明显的发现类似这样
- Start Length Name Class
- 0006:00000000 00000104H LIUAN DATA
的东西,实际地址可能有差别,但是名字不会变。我们要求编译器加入新的符号,它做到了。
在实际使用时,还需要#pragma section事先告诉编译器,我们要使用的段名,完整的示例是这样的:
- #pragma section("LIUAN", read)
- __declspec(allocate("LIUAN")) int myValue = 100;
- int _tmain(int argc, _TCHAR* argv[])
- {
- printf("%d/n",myValue);
- return 0;
- }
- Start Length Name Class
- 0005:000002fc 00000545H .idata$6 DATA
- 0006:00000000 00000104H LIUAN DATA //this line
- 0007:00000000 00000169H .rsrc$01 DATA
__declspec(selectany)告诉编译器我要使用pick-any COMDAT,说白了就是跟编译器说我要在头文件里初始化东西。在头文件里初始化整形数据可以借助enum来模拟实现,但是初始化其他数据就比较难了,selectany给我们提供了一个方式。
完整的使用示例如下:
- //someheader.h
- #pragma section("LIUAN", read)
- extern "C" __declspec(allocate("LIUAN")) __declspec(selectany) int myValue = 100;
- __pragma(comment(linker, "/include:_myValue"))
对于ATL生成的默认com对象,我们打开map文件发现有:
- 0003:0000a344 00000104H LIUAN DATA
对于ATL生成的默认com对象,我们打开map文件发现有
- <p>0003:0000a344 00000104H ATL __a DATA
- 0003:0000a448 00000104H ATL __m DATA
- 0003:0000a54c 00000104H ATL __z DATA</p>
是的,这三个变量就是使用了这种手段写进去的。