我突然发现我们的代码库中有TDataModuleTestExchange(nil)
'构造函数调用':
procedure TDialoogConfigExchange.ButtonTestClick(Sender: TObject);
var
lDataModuleTestExchange: TDataModuleTestExchange;
lResult : Boolean;
begin
inherited;
[snip]
begin
lDataModuleTestExchange := TDataModuleTestExchange(nil); // *** HERE ***
try
lResult := lDataModuleTestExchange.GetCalendarFolder(EditHost.Text,EditGebruiker.Text,EditWachtwoord.Text);
if lResult then
ToonMelding(sExchangeTestGelukt, mtInformation, [mbOk])
else
ToonMelding(Meldingen.Text, mtError, [mbOK]);
finally
lDataModuleTestExchange.Free;
end;
end;
end;
所以代替TDataModuleTestExchange.**Create**(nil)
这样就可以了!
unit dmTestExchange;
interface
uses
System.SysUtils, System.Classes,
Xml.XMLDoc, Xml.XMLIntf, Xml.XMLDOM,
TimeTellDM;
type
TDataModuleTestExchange = class(TTimeTellDataModule) // TDataModule descendant
private
public
function GetCalendarFolder(const AExchangeServerURL,AExchangeLoginName,AExchangePass: String): Boolean;
end;
没有编译器错误,没有运行时问题。怎么样?
答案 0 :(得分:5)
首先,值得指出的是,演员是虚假的,除了混淆之外没有任何其他目的。代码相当于:
lDataModuleTestExchange := nil;
TDataModuleTestExchange.GetCalendarFolder
是一个实例方法,您在nil
引用上调用它。如果方法尝试访问实例中的任何字段,或调用虚方法,或者实际上依赖于实例的任何字段,则会导致运行时错误。因此,TDataModuleTestExchange.GetCalendarFolder
的实现似乎不依赖于实例。虽然你似乎在这里逃避了这一点,但编写这样的代码显然是非常糟糕的形式。
可能会重写该类以声明静态类方法,如下所示:
type
TDataModuleTestExchange = class(TTimeTellDataModule)
public
class function GetCalendarFolder(const AExchangeServerURL,
AExchangeLoginName, AExchangePass: string): Boolean; static;
end;
然后这样叫:
lResult := TDataModuleTestExchange.GetCalendarFolder(EditHost.Text,
EditGebruiker.Text, EditWachtwoord.Text);
if lResult then
ToonMelding(sExchangeTestGelukt, mtInformation, [mbOk])
else
ToonMelding(Meldingen.Text, mtError, [mbOK]);