gpt4 book ai didi

macos - Swift:deinit 方法中没有 println 的输出(不使用 playground)

转载 作者:搜寻专家 更新时间:2023-10-31 22:03:41 24 4
gpt4 key购买 nike

这是我的测试代码(在终端中运行):

#!/usr/bin/xcrun swift

var count = 0; // for reference counting

class A {
init() {
count++;
}
deinit {
println("A deinit")
count--;
}
}

var a: A? = A()
println(count)
a = nil // no output if I comment out this statement
println(count)

输出:

1
A deinit
0

如果上面提到的行被注释掉,则没有输出“A deinit”。输出将是:

1
1

我已经使用 swiftc 编译了代码,但结果还是一样。 (xcrun swiftc -o test test.swift)

是设计使然,程序退出时将关闭标准输出,还是对象在销毁时仍被引用(通过什么?)?

更新:感谢@Logan,现在我有更多关于它的细节。

当它在函数内部运行时,即使我注释掉a = nil,它也会输出A deinit:

#!/usr/bin/xcrun swift

class A {
deinit {
println("A deinit")
}
}

func test() {
var a: A? = A()
//a = nil
}

test()

我没有在 Xcode 中使用 playground。 :-$

更新

#!/usr/bin/xcrun swift

import Foundation

class A {
deinit {
var s = "A deinit"
println(s)

var a: A? = A()
a = nil

var error: NSError?
var path = "\(NSFileManager.defaultManager().currentDirectoryPath)/swift_test.txt"
if s.writeToFile(path, atomically: true, encoding: NSUTF8StringEncoding, error: &error) {
println("File saved at \(path)")
} else {
println(error)
}
}
}

//func test() {
var a: A? = A()
//}
//test()

结果:没有输出到 stdout 或文件,除非在 test 函数中运行。

最佳答案

虽然我没有看到明确的引用说“Swift 的 deinit 与 ObjC 的 dealloc 具有完全相同的语义”,但很难想象这不是事实并非如此,因为 Swift 和 ObjC 对象都由 ARC 管理并且一般来说可以互换。

鉴于此,这完全符合预期。 Cocoa 不会在程序终止时释放对象。它只是终止,泄漏所有内存、文件句柄和其他系统资源,并将其留给操作系统进行清理。这使得程序终止速度比其他方式快得多。

这一点很重要,因为这意味着您通常不应使用 deinit 来管理操作系统管理资源以外的任何资源(当然不是您要求运行的任何资源) .当然,没有办法保证析构函数运行,即使在 C++ 中也是如此。如果你崩溃了,它不会发生,你的程序将不得不处理它。您可以将其视为所有 Cocoa 程序在终止时安静地崩溃。

因此在您的情况下,a = nil 导致 deinit 运行,而程序终止则不会。

关于macos - Swift:deinit 方法中没有 println 的输出(不使用 playground),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27044573/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com