在Delphi 2007中链接多个C对象文件时出错

时间:2011-01-09 07:37:54

标签: c delphi delphi-2007 delphi-xe object-files

我是delphi的新手。我试图在我的Delphi项目中添加C Object文件并直接链接它们,因为Delphi支持C Object Linking。当我链接单个Object文件时,我得到了它。但是当我尝试链接多个目标文件时,我收到错误“不满意的前向或外部声明”。我在Delphi 2007以及XE中尝试了这个。所以我在这里做错了什么?

工作代码:

function a_function():Integer;cdecl;  

implementation  

{$Link 'a.obj'}  

function a_function():Integer;cdecl;external;  

end.

错误代码:

function a_function():Integer;cdecl;  
function b_function();Integer;cdecl;  
function c_function();Integer;cdecl;  

implementation  

 {$LINK 'a.obj'}  
 {$LINK 'b.obj'}  
 {$LINK 'c.obj'}  

function a_function():Integer;cdecl;external;  
function b_function();Integer;cdecl;external;  
function c_function();Integer;cdecl;external;  
end.

1 个答案:

答案 0 :(得分:7)

另外,由@vcldeveloper链接的文章对一些常见问题有很好的解释。在Pascal代码中提供缺少的C RTL函数的技巧非常好,并且比尝试将必要的函数链接为C文件甚至是.obj文件要快得多。

然而,我怀疑我知道这里发生了什么。我使用相同的方法,但实际上在单元中有超过100个.obj文件。我发现当我添加新的时,我会得到与您相同的链接器错误。我解决这个问题的方法是尝试重新订购我的$ LINK指令。我尝试逐个添加新的obj文件,最终我总能解决这个问题。

如果您的C文件完全是独立的,那么您可以将每个文件放在不同的单元中,链接器将处理它。但是,我怀疑是这种情况,而且我怀疑如果它们真的是独立的,那么这个问题就不会发生。此外,最好将$ LINK指令放在一个单元中,以便任何需要提供的RTL功能只能提供一次和一次(它们需要与$ LINK指令出现在同一单元中)。

链接器中的这种奇怪现象存在于Delphi 6中,并且存在于Delphi 2010中。

编辑1 :现在已经意识到这个问题可能是由于Delphi使用单通道编译器。我怀疑“缺少外部引用”错误是因为编译器按照它们出现在单元中的顺序处理.obj文件。

假设a.obj出现在b.obj之前,而a.obj则调用b()b.obj中的函数。编译器不知道b()驻留在需要修复函数调用的位置。当我找到时间时,我会尝试测试这个假设是否至少是合理的!

最后,解决问题的另一个简单方法是将a.c,b.c和c.c合并到一个C文件中,我相信它会绕过OP的这个问题。

编辑2 :我发现了另一个覆盖此问题的Stack Overflow问题:stackoverflow.com/questions/3228127/why-does-the-order-of-linked-object-file-with-l-directive-matter

编辑3 :我找到了解决此问题的另一种真正好方法。每次编译器抱怨

[DCC Error] Unit1.pas(1): E2065 Unsatisfied forward or external declaration: '_a'

你只需在单元的实现部分添加一个类似的声明:

procedure _a; external;

如果您希望从Delphi调用它,那么您显然需要获取参数列表,调用约定等。否则,如果它是外部代码的内部例程,则可以忽略参数列表,调用约定等。

据我所知,这是导入两个以循环方式相互引用的对象的唯一方法。我认为以这种方式宣布外部程序类似于做出前瞻性声明。不同之处在于实现是由对象而不是Pascal代码提供的。

我现在能够为我的军械库添加更多工具 - 谢谢你提出这个问题!