今天在把一个用Visual Studio.NE 2003编写的Managed C++程序改用Visual Studio 2005重新编译的时候遇到了这样的错误:
错误 1 error LNK2001: 无法解析的外部符号 "?.cctor@@$$FYMXXZ" (?.cctor@@$$FYMXXZ) ButtonEncryption.obj
错误 2 fatal error LNK1120: 1 个无法解析的外部命令 F:/JKXDL/资料/PayPal/ButtonEncryptionLib/Debug/ButtonEncryptionLib.dll 1
首先怀疑是C++/CLI语法兼容性的问题,因为自动升级项目后是用/clr:oldSyntax参数编译的。由于项目并不大,就首先将Managed C++的语法全部改成了C++/CLI的语法,并使用/clr参数进行编译,然而令人失望的是,仍然出现了上面的链接错误。
百度后发现优快云上也有问过同样问题的帖子,但是没有解决。后来用Google搜到了解决方法:
http://groups.google.com/group/microsoft.public.dotnet.languages.vc/browse_thread/thread/4d3137c5bb03c6dd/ec322d903925ce88#ec322d903925ce88
里面基本上把这个问题解释清楚了,丢失的符号是void __clrcall __identifier(".cctor")();的托管名称,是模块初始化器,编译器生成的用来初始化全局变量的特殊函数。它是在托管CRT中定义的,即使是项目的代码中没有使用,也会被编译器使用。而VS2005自动升级后的项目属性中的参数/Zl和/NODEFAULTLIB却忽略了对默认库的引用,因此出现了上面的错误。要对项目的属性进行修改,只需要去掉/Zl(编译器) /NOENTRY /NODEFAULTLIB (链接器)这三个参数,并在链接器的“附加依赖项”中去掉对msvcrt.lib的依赖。
还需要注意一点,虽然上面三个编译器选项在项目属性中都有对应的开关,可是我确认把三个开关都设置为正确的值以后,仍然出错。本来以为出现了其他的问题,仔细检查之后,发现在编译器属性的“命令行”页面中有一个附加选项,里面还有一个/Zl,链接器的自定义命令行附加选项中也有一个/NOENTRY,把这两个选项删除,终于编译通过。这是VS.NET 2003中的项目属性直接继承过来的自定义设置,而VS.NET 2003的项目属性中没有对应的开关,只能写在自定义命令行附加选项中。
在此感谢 的解答。