缺少有关变量的警告可能尚未初始化

Posted

技术标签:

【中文标题】缺少有关变量的警告可能尚未初始化【英文标题】:Missing warning about variable might not have initialized 【发布时间】:2015-03-22 18:37:19 【问题描述】:

我的代码中有一个错误。毫无疑问;这是我的错误,完全是我的错:

procedure TfrmCageSetup.actDeleteFloatExecute(Sender: TObject);
var
   conn: TADOConnection;
begin
   ...
   if not float.CanDelete(conn, outnoDeleteReason) then
   begin
   ...
end;

我忘记初始化变量conn。结果,它是堆栈垃圾。当目标被调用者检查参数时,它通过:

function TFloat.CanDelete(Connection: TADOConnection; out NoDeleteReason: string): Boolean;
begin
   if Connection = nil then
      raise EArgumentNullException.Create('Connection');

而且我有时会遇到非常奇怪的访问冲突。

但为什么 Delphi 没有抓住它?

是的,这是我的错。但我使用静态、强类型语言的部分原因是,它可以帮助我发现这些愚蠢的错误。

首先我检查了我实际上并没有将它初始化为一些损坏或破坏的对象(也许它初始化 - 只是很糟糕)。但是不,它确实在声明和首次使用之间未初始化:

然后我想也许我已经关闭了警告:

变量可能尚未初始化

但不,它已启用(全局启用,对于我的平台,对于我的发布类型):

然后我再次构建它(即使这个错误已经在代码中几个月了),以确保我没有错过错误。但不是;除了 VCL 和 JVCL 中的一些错误之外,Delphi 没有发现错误:

我希望 Delphi 帮助我

这是怎么回事?

知道我的代码中是否还有其他地方有同样的、可怕的、可怕的、完全可检测的错误。

也许是 64 位编译器。有人抱怨说 64 位后端(全新)不如 32 位后端智能。如果我尝试将其更改为 32 位怎么办?

还是没有:

与 32 位 release 和 64 位 release 相同。

1/25/2015 - 它甚至显示这样的警告吗?

重现警告的步骤确实显示:

procedure TForm1.FormCreate(Sender: TObject);
var
    silverWarrior: Integer;
begin
    IsWrong(silverWarrior);
end;

function TForm1.IsWrong(n: Integer): Boolean;
begin
    Result := True;
end;

发出警告:

W1036 变量“silverWarrior”可能尚未初始化

您甚至不必将值传递给另一个函数;简单地使用一个未初始化的变量给出警告(取决于编译器那天的心情):

procedure TForm1.FormCreate(Sender: TObject);
var
   silverWarrior: Integer;
   theAnswer: Integer;
begin
   theAnswer := silverWarrior + 42;
end;

给出警告:

W1036 变量“silverWarrior”可能尚未初始化

超过32个局部变量导致失败?

没有;你仍然会收到警告:

procedure TForm1.FormCreate(Sender: TObject);
var
    localVariable1: Integer;
    localVariable2: Integer;
    localVariable3: Integer;
    localVariable4: Integer;
    localVariable5: Integer;
    localVariable6: Integer;
    localVariable7: Integer;
    localVariable8: Integer;
    localVariable9: Integer;
    localVariable10: Integer;
    localVariable11: Integer;
    localVariable12: Integer;
    localVariable13: Integer;
    localVariable14: Integer;
    localVariable15: Integer;
    localVariable16: Integer;
    localVariable17: Integer;
    localVariable18: Integer;
    localVariable19: Integer;
    localVariable20: Integer;
    localVariable21: Integer;
    localVariable22: Integer;
    localVariable23: Integer;
    localVariable24: Integer;
    localVariable25: Integer;
    localVariable26: Integer;
    localVariable27: Integer;
    localVariable28: Integer;
    localVariable29: Integer;
    localVariable30: Integer;
    localVariable31: Integer;
    localVariable32: Integer;
    localVariable33: Integer;
    localVariable34: Integer;
    localVariable35: Integer;
    localVariable36: Integer;
    localVariable37: Integer;
    localVariable38: Integer;
    localVariable39: Integer;
    localVariable40: Integer;
    localVariable41: Integer;
    localVariable42: Integer;
    localVariable43: Integer;
    localVariable44: Integer;
    localVariable45: Integer;
    localVariable46: Integer;
    localVariable47: Integer;
    localVariable48: Integer;
    localVariable49: Integer;
    silverWarrior: Integer;
    theAnswer: Integer;
begin
    theAnswer := silverWarrior + 42;
end;

W1036 变量“silverWarrior”可能尚未初始化

【问题讨论】:

他们似乎做了一些改变,添加了许多以前从未存在过的全新提示和警告,但是这样的事情......让我失去了信心。我们的旧版软件包含数百万行代码,从 XE2 升级到 XE7 时,实际上有超过 5,000 个全新的提示和警告,这些提示和警告是前所未有的。 @Jerry 这听起来很奇怪。我不记得 XE2 到 XE7 中的新提示/警告。 @David 并不是说​​ Delphi 中添加了新的提示/警告,而是由于某些更改,导致它们出现。最常见的是H2443,如上图所示。 @Jerry 为什么这会让你失去信心。他们移动了一些功能。你应该处理这些提示。 @David 是这样的问题让我失去了信心。 Embarcadero 似乎并不关心他们给我们带来的困难。我的观点是在为我们的项目添加数千个新提示和警告的过程中,Ian 的问题指出其他人丢失了。 【参考方案1】:

就是这样。编译器在发现此类错误方面并不完美。除了您的问题(编译器未能发出警告的未初始化变量)之外,很可能会遇到编译器发出警告的初始化变量。更糟糕的是,32 位编译器可能会发出警告,而 64 位编译器不会发出警告。反之亦然,有时 64 位编译器会发出警告,而 32 位编译器则不会。

我没有找到任何模式。我没有解决办法。我有一种暗示,像 breakexit 这样的控制语句可能与这些问题有关,但我没有确凿的证据。

只有 Embarcadero 可以解决这个问题,所以您能做的最好的事情就是提交错误报告并希望 Embarcadero 有人关心。

【讨论】:

巧合的是,Embarcadero 是shutting down QualityCentral。我认为他们想摆脱令人讨厌的积压错误。 我不这么认为。我认为他们想使用更好的错误跟踪系统。我在新系统中提交了许多优秀的错误。 我一开始也是这么想的。但是在搜索 QualityPortal 之后,我发现他们没有转移任何现有的错误。 这可能很难做到。他们不需要将错误转移到新系统。 当 QC 退休时 16 年的突出错误及其变通方法消失了(例如 Ken 的回答中的文章),这将是糟糕透顶【参考方案2】:

我知道至少有 1 种情况可能会丢失此警告。

QC#62702 Compiler should give warning for uninitialized object variables (archive.is)

简而言之,如果将变量作为varout 参数传递给第二个函数,它会禁用整个当前函数的警告。

procedure A
var obj : Tobject;
begin
  obj.Free; <--Warning here
end;

procedure B
var obj : Tobject;
begin
  obj.Free; <--No warning here, even if it's only passed as var/out after this line.
  FreeAndNil(obj);
end;

【讨论】:

哎哟!!这绝对是个臭屁!【参考方案3】:

另一个因素:Delphi 的变量跟踪中有一个不会修复的“错误”。如果你有超过 32 个局部变量,它会很糟糕。由于那不是合理的代码,他们没有做任何事情。

你能打到这个吗?

【讨论】:

方法中有超过32个局部变量?因为,不,只有四个。整个项目超过32个局部变量?绝对地。但即使方法中有超过 32 个局部变量,我仍然会收到警告。 @IanBoyd 好的。我不知道这是整个代码还是只是相关的部分。

以上是关于缺少有关变量的警告可能尚未初始化的主要内容,如果未能解决你的问题,请参考以下文章

延迟初始化健全性检查:变量可能尚未初始化

错误:变量可能尚未初始化 - 如何初始化ArrayList?

可能尚未为Cursor android初始化局部变量

如何初始化比较器?它说变量可能尚未初始化[重复]

如何修复警告:初始化程序周围缺少大括号?

如何避免 MSVC 警告 C4701:可能未初始化的局部变量