假设下面情形:
某DLL为客户代码导出了下面类
class __declspec(dllexport) CTest
{
public :
void print1()
{
printf("print1");
c[1023] = 0xEEEEEEEE ;
}
void print2()
{
printf("print2");
}
int m_a;
};
客户端在使用该DLL时,如下声明
class __declspec(dllimport) CTest
{
public :
void print1();
void print2();
int m_a;
};
此时客户端使用该DLL导出类一切正常。
为了功能升级,假设 DLL导出类有变化:
class __declspec(dllexport) CTest
{
public :
void print1()
{
printf("print1");
miData[1023] = 0xEEEEEEEE ;
}
void print2()
{
printf("print2");
}
private:
int m_a;
int miData[1024];
};
用新版本的DLL 直接替换客户端正在使用的旧版本的DLL后,客户端的应用程序将会出错(内存访问违规)。
原因是: 在客户端的EXE 中为CTest 分配了4 个字节的空间,当直接替换DLL后,位于客户EXE模块的分配堆空间的代码是不变的,仍然是4个字节。此时新版本的DLL的print1 必然会出现内存访问出错。
解决上面问题的方法是: DLL 只能导出接口类, 并且创建对象的代码要在 DLL 模块完成,客户端只访问提供的接口。