【问题标题】:.tlh generated on 2 machines is different在 2 台机器上生成的 .tlh 不同
【发布时间】:2010-10-29 00:40:59
【问题描述】:

我有一个 .NET dll,它有一些暴露给 com 的接口\类。在构建过程中会生成一个 .tlb 文件,并且该 tlb 被一些 c++ 代码引用。因此,编译器会为 tlb 生成一个 .tlh 文件。

当我在本地运行构建时,其中一个接口中的属性之一最终会在 tlh 中具有不同名称的相应方法。 .net 代码中的属性称为 PropertyA,最终称为 get_propertyA,而 PropertyB 最终称为 get_PropertyB。发生这种情况时,我没有眨眼,只是使用了 tlh 中定义的方法并假设一切都是 hunky dory,但是当我提交这些更改时,构建对其他人不起作用,因为编译器生成了名为 get_PropertyA 的属性和get_PropertyB(注意 propertyA 中的大小写不匹配)。

在两台机器上生成的 tlb 文件是相同的(根据十六进制比较器),并且 tlh 文件都是由相同的编译器版本生成的。

构建过程通过执行以下操作创建 tlb:regasm path\to\dll\Mydll.dll -tlb:path\to\output\mydll.tlb

任何想法为什么我的本地版本最终会出现名称不正确的属性?或者我能做些什么来解决它?

更新:我读到 tlbexp 将使用它找到的字符串的第一个版本,并且可以通过重新编译来改变。虽然我没有使用 tlbexp,但我想知道这是否是问题所在。我找到了与我的方法(在其他方法中)同名的参数,但开头有一个小写字母。所以我把这些都换了。重建,没有变化。所以我然后重命名了我的 COM 方法。重新构建并得到预期的缺失方法错误。将该方法重命名为原始名称,嘿,它似乎已修复。由于它现在似乎可以工作并且我不能让它再次失败,所以我无法尝试建议的解决方案,但我喜欢重命名的想法,以防将来发生这种情况。

【问题讨论】:

  • 使用“重命名”解决方法时要小心 - 它的工作方式类似于简单的文本替换,有时会导致奇怪的结果。我已经更新了答案以概述这一点。
  • 感谢您的提醒。如果出现问题,我会记住这一点。

标签: .net compiler-construction com interop typelib


【解决方案1】:

我也遇到了同样的问题。

通过另一个 SO 问题 (https://stackoverflow.com/questions/708721/compare-type-libraries-generated-by-tlbexp) 我发现了这个社区内容:

http://social.msdn.microsoft.com/Forums/en-US/clr/thread/5003c486-ed3f-4ec8-8398-a1251b0f9e74

引用该内容:

在 tlbexp 的文档中,有一个有用的社区内容:

http://msdn2.microsoft.com/en-gb/library/hfzzah2c(VS.80).aspx

引用:

“使用 /names 选项的原因是类型库将每个标识符存储在不区分大小写的表中。遇到的第一个案例获胜。因此,如果有一个参数,名为 Monitor 的类最终可能会暴露为“monitor”首先遇到这样的名称。(遇到标识符的顺序可以通过重新编译程序集来改变!) /names 可以保证稳定的大小写。"

根本原因似乎是中间的一个错误,如下所述:

http://support.microsoft.com/default.aspx?scid=kb;en-us;220137

引用:

“当有两个仅大小写不同的标识符时,第二个标识符的大小写将更改以反映第一个标识符的大小写。”

因此,作为解决方案,我在项目设置中取消选中“注册 COM 互操作”选项并添加了构建后步骤

"$(DevEnvDir)....\SDK\v2.0\Bin\tlbexp" $(TargetFileName) /names:"$(ProjectDir)Names.txt" %windir%\Microsoft.NET\Framework\v2.0.50727\regasm $(TargetFileName)

名称文件包含定义如何进行大写的条目。就我而言,它只包含一行:

身份证

最好的问候

伯恩德·里特

使用 /names 为我解决了这个问题。

【讨论】:

  • 感谢更新,尽管问题很老。为另一种方法 +1。
【解决方案2】:

您可以使用 importrename 属性来显式重命名属性。假设你有propA,有时会变成PropApropB,有时会变成PropB。要始终拥有PropAPropB,请使用 rename,如下所示:

#import <library> rename( "propA", "PropA" ) rename( "propB", "PropB" )

小心使用它——它会导致一个简单的文本替换,它适用于它在类型库中遇到的任何标识符。在某些情况下,它可能会导致难以调试不需要的副作用。

【讨论】:

  • 无法测试以查看这是否解决了问题(请参阅上面有问题的更新),但似乎它可能无法接受作为答案,但我已赞成。如果问题再次出现,我会试试这个并回来。感谢您的建议
  • 好的,问题又来了。我对此进行了测试,似乎效果很好。谢谢!
  • 我没有投反对票,因为这确实有效,但这会迫使您在每次导入库时都进行重命名。首先强制正确创建 tlb 不是更好吗(参见 tlbexp 上的 /names)
【解决方案3】:

完整性检查:您确定在两台机器上使用相同的#import 指令吗?即正在编译的完全相同的源文件?

尝试在包含项目的目录上创建网络共享并在另一台机器上打开它,以便 1000% 确定这些是正在编译的相同源文件。

抱歉,我没有更具体的建议。

【讨论】:

  • 是的,我们已经通过了健全性检查。正在编译相同的文件,我们检查了 cvs 版本号。并且我们检查了中间步骤是否生成了正确的文件(tlb 文件是相同的)并且属性是在这个版本中新引入(重命名)的,所以它们出现的事实表明它使用了正确的源。 ..
  • 您可以发布您正在使用的#import 行吗?
  • #import raw_interfaces_only,raw_native_types,no_namespace,named_guids
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2020-02-12
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2015-11-17
  • 1970-01-01
  • 2013-10-23
相关资源
最近更新 更多