- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有相同代码的两个变体:
{$APPTYPE CONSOLE}
uses
System.SysUtils;
type
IMyObject1 = interface
['{4411181F-3531-4D30-AB18-A8326F8C2CD0}']
end;
IMyObject2 = interface
['{41C88E1A-0360-4AC3-B021-125880B23DE5}']
end;
TMyObject = class(TInterfacedObject, IMyObject1, IMyObject2)
public
destructor Destroy; override;
end;
destructor TMyObject.Destroy;
begin
Writeln('Destroy');
inherited;
end;
procedure Variant1;
function GetMyObject: IMyObject2;
var
Obj1: IMyObject1;
begin
Obj1 := TMyObject.Create;
try
Obj1.QueryInterface(IMyObject2, Result);
finally
Obj1 := nil;
end;
end;
var
Obj2: IMyObject2;
begin
Obj2 := GetMyObject;
try
finally
Obj2 := nil;
end;
Writeln('Variant1 end of proc');
end;
function GetMyObject: IMyObject2;
var
Obj1: IMyObject1;
begin
Obj1 := TMyObject.Create;
try
Obj1.QueryInterface(IMyObject2, Result);
finally
Obj1 := nil;
end;
end;
procedure Variant2;
var
Obj2: IMyObject2;
begin
Obj2 := GetMyObject;
try
finally
Obj2 := nil;
end;
Writeln('Variant2 end of proc');
end;
begin
Variant1;
Writeln('---');
Variant2;
Writeln('---');
Readln;
end.
输出
Variant1 end of procDestroy---DestroyVariant2 end of proc---
为什么两个变体的行为不同?
最佳答案
测试环境:Delphi XE7 Windows编译器
在Variant1
中,编译器决定需要创建一个隐式局部变量来保存额外的接口(interface)引用。我无法辨别为什么会这样。但这里发出的代码表明这就是发生的情况:
变体1
Project1.dpr.46: Obj2 := GetMyObject;0041A3BD 55 push ebp0041A3BE 8D45F8 lea eax,[ebp-$08]0041A3C1 E836FFFFFF call GetMyObject0041A3C6 59 pop ecx0041A3C7 8B55F8 mov edx,[ebp-$08]0041A3CA 8D45FC lea eax,[ebp-$04]0041A3CD E816F5FEFF call @IntfCopy // copies into the implicit local
变体2
Project1.dpr.70: Obj2 := GetMyObject;0041A53B 8D45FC lea eax,[ebp-$04]0041A53E E839FFFFFF call GetMyObject // no such copy here
隐式局部变量在声明它们的函数的最后完成,就像任何其他局部变量一样。这就是为什么 Writeln('Variant1 end of proc')
执行,然后隐式局部被最终确定,然后释放对该对象的最终引用。
这里非常有趣的一个因素是,如果您启用优化,那么输出将更改为:
DestroyVariant1 end of proc---DestroyVariant2 end of proc---
出于某种原因,编译器决定在启用优化时不创建隐式局部变量。
当然,以上都是针对32位编译器的。 64位编译器又不一样了,你难道不知道吗?在 64 位编译器中,无论优化如何,输出都是按照问题进行的。
我认为这个问题可能与我的问题Is the compiler treatment of implicit interface variables documented?有关据我所知,没有关于编译器功能的官方规范或文档,您在这里所能做的最好的事情就是通过观察其输出来进行经验学习。优化会改变行为意味着您应该尝试避免容易受到此类行为变化影响的代码。如果你确实可以预测它们的话。
关于delphi - 最后一次 IUnknown.Release 调用的时刻,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31108871/
为什么 Visual Studio 编译器满意 void fn(int *&i) { ; } 和 void fn(IUnknown *const &p) { ; } 但不是 void
如果我有 IUnknown *ptr , 我需要调用Release()在我通过ptr->QueryInterface()获得的每个界面上, 另外调用ptr->Release()当我完成 ptr ? 我
使用接口(interface)时的大量示例,例如 IUnknown ,在本例中为 IDocHostUIHandler但这并不重要 - 使用与此类似的代码: class TDocHostUIHandl
我正在尝试使用带有 ATL 的 COM 聚合来实现共享逻辑。我定义了一个基类,称为CameraBase ,这只能通过聚合获得。为此,我添加了 aggregateable注释为 coclass -声明。
我需要将一个未知类型的变量转换为另一个类型type(SpaceClaim::Api::V10::Session^) 我正在使用 c/cli,我找不到任何解决方案 修复这个 prb 的代码是: gcr
我正在尝试创建一个实现 IUnknown 接口(interface)的类。我在头文件中有以下代码: #pragma once #include "stdafx.h" #include "Unknwn.
这是实现 IUnknown COM 的Release 方法的标准(不是推荐)方法界面(直接取自MSDN): ULONG CMyMAPIObject::Release() { // Decrem
我有相同代码的两个变体: {$APPTYPE CONSOLE} uses System.SysUtils; type IMyObject1 = interface ['{4411181
当我使用类型库导入程序将特定于 .net dll 的 tlb 文件导入 Delphi 时,接受 .net 特定类型参数的方法被 IUnknown 替换。当我想从我的 Delphi 客户端应用程序调用这
我是 COM 新手。为什么接口(interface)方法 QueryInterface AddRef 和 Release 被声明为具有 __stdcall 而不是任何其他调用(__cdlecl、__t
为什么在析构函数中对 IWICImagingFactory 对象调用继承的 IUnknown::Release() 函数会导致“CXX0030:错误:无法计算表达式”显示在对象的虚函数表 (__vfp
ATL END_COM_MAP宏定义如下: #define END_COM_MAP() \ __if_exists(_GetAttrEntries) {{NULL, (DWORD_PTR)_G
如何在运行时检查 void* 是否为 IUnknown*? IUnknown *unk = dynamic_cast(item); 不工作(编译错误)。 最佳答案 您无法在运行时明确地确定这一点。您无
我在 VB6 中有一个属性,我正试图将其转换为 C#。具体如下: Public Property Get NewEnum() As IUnknown 'this property allows
我一直在寻找如何实现 IUnknown 的示例在 C# 中,但还没有找到任何合适的引用或解决方案。 是否应该像...一样简单 public interface IUnknown { UInt3
我正在 Delphi 中创建一个 ActiveX 服务器来替换在 Delphi 中创建的 native DLL,以便可以在 .NET 中调用它。 (我是 Delphi 新手,更喜欢 C#) 原始 DL
我从类型库(硬件 SDK 的一部分)导入的 COM 接口(interface)的一些方法返回或接收 IUnknown 类型的值。例如,SDK文档指定的方法如下: bool SetInput1Selec
在标题中,包含以下代码。 inline void SafeRelease( IUnknown * & in_COM_Pointer ) { if ( NULL != in_CO
从例子中我看到 COM IUnknown::Release() 函数实现是这样的: ULONG Release() { InterlockedDecrement(&m_count);
我写了两个小的 C++ 应用程序,其中一个是嵌入了一些 ActiveX 控件的 ActiveX 容器。此容器应用程序知道引用 ActiveX 控件的 IUnknown*。 另一个应用程序是一个客户端,
我是一名优秀的程序员,十分优秀!