- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
Delphi 是否在对象完全构造之前分配实例变量?
换句话说,给定一个变量:
var
customer: TCustomer = nil;
然后我们构建一个客户并将其分配给变量:
customer := TCustomer.Create;
是否有可能customer
不能为nil
,但不能指向完全构造的TCustomer
?
执行延迟初始化时,这会成为一个问题:
function SacrifialCustomer: TCustomer;
begin
if (customer = nil) then
begin
criticalSection.Enter;
try
customer := TCustomer.Create;
finally
criticalSection.Leave;
end;
end;
Result := customer;
end;
错误所在行:
if (customer = nil)
有可能另一个线程调用:
customer := TCustomer.Create;
and the variable is assigned a value before construction happens 。这会导致线程假设 customer
是一个有效对象,因为变量已被分配。
这个多线程单例错误会在 Delphi (5) 中发生吗?
<小时/>奖励问题
是否有一个可接受的、线程安全的one-time initialization Delphi 的设计模式?许多人通过重写NewInstance
和FreeInstance
在Delphi中实现了单例;它们的实现将在多个线程中失败。
严格来说,我并不是在寻找如何实现和单例的答案,而是延迟初始化。虽然单例可以使用延迟初始化,但延迟初始化并不限于单例。
更新
两个人建议了一个答案 that contains a common mistake. The broken double-checked locking algorithm translated to Delphi :
// Broken multithreaded version
// "Double-Checked Locking" idiom
if (customer = nil) then
begin
criticalSection.Enter;
try
if (customer = nil) then
customer := TCustomer.Create;
finally
criticalSection.Leave;
end;
end;
Result := customer;
来自Wikipedia :
<小时/>Intuitively, this algorithm seems like an efficient solution to the problem. However, this technique has many subtle problems and should usually be avoided.
另一个错误建议:
function SacrificialCustomer: TCustomer;
var
tempCustomer: TCustomer;
begin
tempCustomer = customer;
if (tempCustomer = nil) then
begin
criticalSection.Enter;
try
if (customer = nil) then
begin
tempCustomer := TCustomer.Create;
customer := tempCustomer;
end;
finally
criticalSection.Leave;
end;
end;
Result := customer;
end;
更新
我创建了一些代码并查看了 cpu 窗口。看来这个编译器,使用我的优化设置,在这个版本的 Windows 上,使用这个对象,首先构造对象,然后分配变量:
customer := TCustomer.Create;
mov dl,$01
mov eax,[$0059d704]
call TCustomer.Create
mov [customer],eax;
Result := customer;
mov eax,[customer];
当然,我不能说保证总是这样。
最佳答案
我对你的问题的解读是你在问这个:
How can I, using Delphi 5 targeting x86 hardware, implement thread-safe lazy initialization of a singleton.
据我所知,您有三种选择。
<强>1。使用锁
function GetCustomer: TCustomer;
begin
Lock.Acquire;
try
if not Assigned(Customer) then // Customer is a global variable
Customer := TCustomer.Create;
Result := Customer;
finally
Lock.Release;
end;
end;
这样做的缺点是,如果 GetCustomer
上存在争用,那么锁的序列化将抑制扩展。我怀疑人们对此的担心超出了必要的范围。例如,如果您有一个执行大量工作的线程,则该线程可以获取对单例的引用的本地副本以减少争用。
procedure ThreadProc;
var
MyCustomer: TCustomer;
begin
MyCustomer := GetCustomer;
// do lots of work with MyCustomer
end;
<强>2。双重检查锁定
这种技术允许您在创建单例后避免锁争用。
function GetCustomer: TCustomer;
begin
if Assigned(Customer) then
begin
Result := Customer;
exit;
end;
Lock.Acquire;
try
if not Assigned(Customer) then
Customer := TCustomer.Create;
Result := Customer;
finally
Lock.Release;
end;
end;
双重检查锁定是一种有着相当复杂历史的技术。最著名的讨论是The "Double-Checked Locking is Broken" Declaration 。这主要是在 Java 上下文中设置的,所描述的问题不适用于您的情况(Delphi 编译器、x86 硬件)。确实,对于 Java 来说,随着 JDK5 的出现,我们现在可以说 Double-Checked Locking 已经被修复了。
Delphi 编译器不会根据对象的构造重新排序对单例变量的写入。此外,强大的 x86 内存模型意味着处理器重新排序不会破坏这一点。请参阅Who ordered memory fences on an x86?
简单地说,双重检查锁定在 Delphi x86 上不会被破坏。更重要的是,x64 内存模型也很强大,并且双重检查锁定也没有被破坏。
<强>3。比较和交换
如果您不介意创建单例类的多个实例,然后丢弃除一个之外的所有实例,则可以使用比较和交换。 VCL 的最新版本利用了这种技术。它看起来像这样:
function GetCustomer;
var
LCustomer: TCustomer;
begin
if not Assigned(Customer) then
begin
LCustomer := TCustomer.Create;
if InterlockedCompareExchangePointer(Pointer(Customer), LCustomer, nil) <> nil then
LCustomer.Free;
end;
Result := Customer;
end;
关于delphi - Delphi 是否在构造对象之前分配变量?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10805769/
我的一位教授给了我们一些考试练习题,其中一个问题类似于下面(伪代码): a.setColor(blue); b.setColor(red); a = b; b.setColor(purple); b
我似乎经常使用这个测试 if( object && object !== "null" && object !== "undefined" ){ doSomething(); } 在对象上,我
C# Object/object 是值类型还是引用类型? 我检查过它们可以保留引用,但是这个引用不能用于更改对象。 using System; class MyClass { public s
我在通过 AJAX 发送 json 时遇到问题。 var data = [{"name": "Will", "surname": "Smith", "age": "40"},{"name": "Wil
当我尝试访问我的 View 中的对象 {{result}} 时(我从 Express js 服务器发送该对象),它只显示 [object][object]有谁知道如何获取 JSON 格式的值吗? 这是
我有不同类型的数据(可能是字符串、整数......)。这是一个简单的例子: public static void main(String[] args) { before("one"); }
嗨,我是 json 和 javascript 的新手。 我在这个网站找到了使用json数据作为表格的方法。 我很好奇为什么当我尝试使用 json 数据作为表时,我得到 [Object,Object]
已关闭。此问题需要 debugging details 。目前不接受答案。 编辑问题以包含 desired behavior, a specific problem or error, and the
我听别人说 null == object 比 object == null check 例如: void m1(Object obj ) { if(null == obj) // Is thi
Match 对象 提供了对正则表达式匹配的只读属性的访问。 说明 Match 对象只能通过 RegExp 对象的 Execute 方法来创建,该方法实际上返回了 Match 对象的集合。所有的
Class 对象 使用 Class 语句创建的对象。提供了对类的各种事件的访问。 说明 不允许显式地将一个变量声明为 Class 类型。在 VBScript 的上下文中,“类对象”一词指的是用
Folder 对象 提供对文件夹所有属性的访问。 说明 以下代码举例说明如何获得 Folder 对象并查看它的属性: Function ShowDateCreated(f
File 对象 提供对文件的所有属性的访问。 说明 以下代码举例说明如何获得一个 File 对象并查看它的属性: Function ShowDateCreated(fil
Drive 对象 提供对磁盘驱动器或网络共享的属性的访问。 说明 以下代码举例说明如何使用 Drive 对象访问驱动器的属性: Function ShowFreeSpac
FileSystemObject 对象 提供对计算机文件系统的访问。 说明 以下代码举例说明如何使用 FileSystemObject 对象返回一个 TextStream 对象,此对象可以被读
我是 javascript OOP 的新手,我认为这是一个相对基本的问题,但我无法通过搜索网络找到任何帮助。我是否遗漏了什么,或者我只是以错误的方式解决了这个问题? 这是我的示例代码: functio
我可以很容易地创造出很多不同的对象。例如像这样: var myObject = { myFunction: function () { return ""; } };
function Person(fname, lname) { this.fname = fname, this.lname = lname, this.getName = function()
任何人都可以向我解释为什么下面的代码给出 (object, Object) 吗? (console.log(dope) 给出了它应该的内容,但在 JSON.stringify 和 JSON.parse
我正在尝试完成散点图 exercise来自免费代码营。然而,我现在只自己学习了 d3 几个小时,在遵循 lynda.com 的教程后,我一直在尝试确定如何在工具提示中显示特定数据。 This code
我是一名优秀的程序员,十分优秀!