我是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.
答案 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代码提供的。
我现在能够为我的军械库添加更多工具 - 谢谢你提出这个问题!